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

com.browserup.bup.BrowserUpProxyServer Maven / Gradle / Ivy

There is a newer version: 3.0.1
Show newest version
package com.browserup.bup;

import com.browserup.bup.assertion.HarEntryAssertion;
import com.browserup.bup.assertion.ResponseTimeLessThanOrEqualAssertion;
import com.browserup.bup.assertion.error.HarEntryAssertionError;
import com.browserup.bup.assertion.field.content.ContentContainsStringAssertion;
import com.browserup.bup.assertion.field.content.ContentDoesNotContainStringAssertion;
import com.browserup.bup.assertion.field.content.ContentMatchesAssertion;
import com.browserup.bup.assertion.field.content.ContentSizeLessThanOrEqualAssertion;
import com.browserup.bup.assertion.field.header.FilteredHeadersContainStringAssertion;
import com.browserup.bup.assertion.field.header.FilteredHeadersDoNotContainStringAssertion;
import com.browserup.bup.assertion.field.header.FilteredHeadersMatchAssertion;
import com.browserup.bup.assertion.field.header.HeadersContainStringAssertion;
import com.browserup.bup.assertion.field.header.HeadersDoNotContainStringAssertion;
import com.browserup.bup.assertion.field.header.HeadersMatchAssertion;
import com.browserup.bup.assertion.field.status.StatusBelongsToClassAssertion;
import com.browserup.bup.assertion.field.status.StatusEqualsAssertion;
import com.browserup.bup.assertion.model.AssertionEntryResult;
import com.browserup.bup.assertion.model.AssertionResult;
import com.browserup.bup.assertion.supplier.CurrentStepHarEntriesSupplier;
import com.browserup.bup.assertion.supplier.HarEntriesSupplier;
import com.browserup.bup.assertion.supplier.MostRecentHarEntrySupplier;
import com.browserup.bup.assertion.supplier.MostRecentUrlFilteredHarEntrySupplier;
import com.browserup.bup.assertion.supplier.UrlFilteredHarEntriesSupplier;
import com.browserup.bup.client.ClientUtil;
import com.browserup.bup.filters.AddHeadersFilter;
import com.browserup.bup.filters.AutoBasicAuthFilter;
import com.browserup.bup.filters.BlocklistFilter;
import com.browserup.bup.filters.BrowserUpHttpFilterChain;
import com.browserup.bup.filters.HarCaptureFilter;
import com.browserup.bup.filters.HttpConnectHarCaptureFilter;
import com.browserup.bup.filters.HttpsHostCaptureFilter;
import com.browserup.bup.filters.HttpsOriginalHostCaptureFilter;
import com.browserup.bup.filters.LatencyFilter;
import com.browserup.bup.filters.RegisterRequestFilter;
import com.browserup.bup.filters.RequestFilter;
import com.browserup.bup.filters.RequestFilterAdapter;
import com.browserup.bup.filters.ResolvedHostnameCacheFilter;
import com.browserup.bup.filters.ResponseFilter;
import com.browserup.bup.filters.ResponseFilterAdapter;
import com.browserup.bup.filters.RewriteUrlFilter;
import com.browserup.bup.filters.UnregisterRequestFilter;
import com.browserup.bup.filters.AllowlistFilter;
import com.browserup.bup.mitm.KeyStoreFileCertificateSource;
import com.browserup.bup.mitm.TrustSource;
import com.browserup.bup.mitm.keys.ECKeyGenerator;
import com.browserup.bup.mitm.keys.RSAKeyGenerator;
import com.browserup.bup.mitm.manager.ImpersonatingMitmManager;
import com.browserup.bup.proxy.ActivityMonitor;
import com.browserup.bup.proxy.BlocklistEntry;
import com.browserup.bup.proxy.CaptureType;
import com.browserup.bup.proxy.RewriteRule;
import com.browserup.bup.proxy.Allowlist;
import com.browserup.bup.proxy.auth.AuthType;
import com.browserup.bup.proxy.dns.AdvancedHostResolver;
import com.browserup.bup.proxy.dns.DelegatingHostResolver;
import com.browserup.bup.util.BrowserUpHttpUtil;
import com.browserup.bup.util.BrowserUpProxyUtil;
import com.browserup.bup.util.HttpStatusClass;
import com.browserup.harreader.model.Har;
import com.browserup.harreader.model.HarCreatorBrowser;
import com.browserup.harreader.model.HarEntry;
import com.browserup.harreader.model.HarLog;
import com.browserup.harreader.model.HarPage;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.MapMaker;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.http.HttpHeaderNames;
import io.netty.handler.codec.http.HttpObject;
import io.netty.handler.codec.http.HttpRequest;
import org.apache.commons.lang3.StringUtils;
import org.littleshoot.proxy.ChainedProxyAdapter;
import org.littleshoot.proxy.ChainedProxyManager;
import org.littleshoot.proxy.HttpFilters;
import org.littleshoot.proxy.HttpFiltersSource;
import org.littleshoot.proxy.HttpFiltersSourceAdapter;
import org.littleshoot.proxy.HttpProxyServer;
import org.littleshoot.proxy.HttpProxyServerBootstrap;
import org.littleshoot.proxy.MitmManager;
import org.littleshoot.proxy.extras.SelfSignedSslEngineSource;
import org.littleshoot.proxy.impl.DefaultHttpProxyServer;
import org.littleshoot.proxy.impl.ProxyUtils;
import org.littleshoot.proxy.impl.ThreadPoolConfiguration;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.net.ssl.SSLEngine;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
import java.util.EnumSet;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import java.util.regex.Pattern;

import static java.util.stream.Collectors.toCollection;

/**
 * A LittleProxy-based implementation of {@link BrowserUpProxy}.
 */
public class BrowserUpProxyServer implements BrowserUpProxy {
    private static final Logger log = LoggerFactory.getLogger(BrowserUpProxyServer.class);
    private static final Object LOCK = new Object();
    private static final Object GET_HAR_LOCK = new Object();

    public static final String DEFAULT_PAGE_REF = "Default";
    public static final String DEFAULT_PAGE_TITLE = "Default";

    private static final HarCreatorBrowser HAR_CREATOR_VERSION =
        new HarCreatorBrowser();
    static {
        HAR_CREATOR_VERSION.setName("BrowserUp Proxy");
        HAR_CREATOR_VERSION.setVersion(BrowserUpProxyUtil.getVersionString());
    }

    /* Default MITM resources */
    private static final String RSA_KEYSTORE_RESOURCE = "/sslSupport/ca-keystore-rsa.p12";
    private static final String EC_KEYSTORE_RESOURCE = "/sslSupport/ca-keystore-ec.p12";
    private static final String KEYSTORE_TYPE = "PKCS12";
    private static final String KEYSTORE_PRIVATE_KEY_ALIAS = "key";
    private static final String KEYSTORE_PASSWORD = "password";

    /**
     * The default pseudonym to use when adding the Via header to proxied requests.
     */
    public static final String VIA_HEADER_ALIAS = "BrowserUpProxy";

    /**
     * True only after the proxy has been successfully started.
     */
    private final AtomicBoolean started = new AtomicBoolean(false);

    /**
     * True only after the proxy has been successfully started, then successfully stopped or aborted.
     */
    private final AtomicBoolean stopped = new AtomicBoolean(false);

    /**
     * Tracks the current page count, for use when auto-generating HAR page names.
     */
    private final AtomicInteger harPageCount = new AtomicInteger(0);

    /**
     * When true, MITM will be disabled. The proxy will no longer intercept HTTPS requests, but they will still be proxied.
     */
    private volatile boolean mitmDisabled = false;

    /**
     * The MITM manager that will be used for HTTPS requests.
     */
    private volatile MitmManager mitmManager;

    /**
     * The list of filterFactories that will generate the filters that implement BrowserUp-proxy behavior.
     */
    private final List filterFactories = new CopyOnWriteArrayList<>();

    /**
     * List of rejected URL patterns
     */
    private volatile Collection blocklistEntries = new CopyOnWriteArrayList<>();

    /**
     * List of URLs to rewrite
     */
    private volatile CopyOnWriteArrayList rewriteRules = new CopyOnWriteArrayList<>();

    /**
     * The LittleProxy instance that performs all proxy operations.
     */
    private volatile HttpProxyServer proxyServer;

    /**
     * No capture types are enabled by default.
     */
    private volatile EnumSet harCaptureTypes = EnumSet.noneOf(CaptureType.class);

    /**
     * The current HAR being captured.
     */
    private volatile Har har;
    /**
     * The current HarPage to which new requests will be associated.
     */
    private volatile HarPage currentHarPage;
    /**
     * Maximum bandwidth to consume when reading responses from servers.
     */
    private volatile long readBandwidthLimitBps;
    /**
     * Maximum bandwidth to consume when writing requests to servers.
     */
    private volatile long writeBandwidthLimitBps;
    /**
     * List of accepted URL patterns. Unlisted URL patterns will be rejected with the response code contained in the Allowlist.
     */
    private final AtomicReference allowlist = new AtomicReference<>(Allowlist.ALLOWLIST_DISABLED);

    /**
     * Additional headers that will be sent with every request. The map is declared as a ConcurrentMap to indicate that writes may be performed
     * by other threads concurrently (e.g. due to an incoming REST call), but the concurrencyLevel is set to 1 because modifications to the
     * additionalHeaders are rare, and in most cases happen only once, at start-up.
     */
    private volatile ConcurrentMap additionalHeaders = new MapMaker().concurrencyLevel(1).makeMap();

    /**
     * The amount of time to wait while connecting to a server.
     */
    private volatile int connectTimeoutMs;

    /**
     * The amount of time a connection to a server can remain idle while receiving data from the server.
     */
    private volatile int idleConnectionTimeoutSec;

    /**
     * The amount of time to wait before forwarding the response to the client.
     */
    private volatile int latencyMs;

    /**
     * Set to true once the HAR capture filter has been added to the filter chain.
     */
    private final AtomicBoolean harCaptureFilterEnabled = new AtomicBoolean(false);

    /**
     * Set to true when LittleProxy has been bootstrapped with the default chained proxy. This allows modifying the chained proxy
     * after the proxy has been started.
     */
    private final AtomicBoolean bootstrappedWithDefaultChainedProxy = new AtomicBoolean(false);

    /**
     * The address of an upstream chained proxy to route traffic through.
     */
    private volatile InetSocketAddress upstreamProxyAddress;

    /**
     * Whether to connect to that upstream chained proxy using https, rather than http
     */
    private volatile boolean upstreamProxyHTTPS;

    /**
     * The hosts and addresses that should not be routed through the upstream proxy
     */
    private List upstreamProxyNonProxyHosts;

    /**
     * The chained proxy manager that manages upstream proxies.
     */
    private volatile ChainedProxyManager chainedProxyManager;

    /**
     * The address of the network interface from which the proxy will initiate connections.
     */
    private volatile InetAddress serverBindAddress;

    /**
     * The TrustSource that will be used to validate servers' certificates. If null, will not validate server certificates.
     */
    private volatile TrustSource trustSource = TrustSource.defaultTrustSource();

    /**
     * When true, use Elliptic Curve keys and certificates when impersonating upstream servers.
     */
    private volatile boolean useEcc = false;

    /**
     * Resolver to use when resolving hostnames to IP addresses. This is a bridge between {@link org.littleshoot.proxy.HostResolver} and
     * {@link com.browserup.bup.proxy.dns.AdvancedHostResolver}. It allows the resolvers to be changed on-the-fly without re-bootstrapping the
     * littleproxy server. The default resolver (native JDK resolver) can be changed using {@link #setHostNameResolver(com.browserup.bup.proxy.dns.AdvancedHostResolver)} and
     * supplying one of the pre-defined resolvers in {@link ClientUtil}, such as {@link ClientUtil#createDnsJavaWithNativeFallbackResolver()}
     * or {@link ClientUtil#createDnsJavaResolver()}. You can also build your own resolver, or use {@link com.browserup.bup.proxy.dns.ChainedHostResolver}
     * to chain together multiple DNS resolvers.
     */
    private final DelegatingHostResolver delegatingResolver = new DelegatingHostResolver(ClientUtil.createNativeCacheManipulatingResolver());

    private final ActivityMonitor activityMonitor = new ActivityMonitor();

    /**
     * The acceptor and worker thread configuration for the Netty thread pools.
     */
    private volatile ThreadPoolConfiguration threadPoolConfiguration;

    /**
     * A mapping of hostnames to base64-encoded Basic auth credentials that will be added to the Authorization header for
     * matching requests.
     */
    private final ConcurrentMap basicAuthCredentials = new MapMaker()
            .concurrencyLevel(1)
            .makeMap();

    /**
     * Base64-encoded credentials to use to authenticate with the upstream proxy.
     */
    private volatile String chainedProxyCredentials;

    public BrowserUpProxyServer() {
    }

    @Override
    public void start(int port, InetAddress clientBindAddress, InetAddress serverBindAddress) {
        boolean notStarted = started.compareAndSet(false, true);
        if (!notStarted) {
            throw new IllegalStateException("Proxy server is already started. Not restarting.");
        }

        InetSocketAddress clientBindSocket;
        if (clientBindAddress == null) {
            // if no client bind address was specified, bind to the wildcard address
            clientBindSocket = new InetSocketAddress(port);
        } else {
            clientBindSocket = new InetSocketAddress(clientBindAddress, port);
        }

        this.serverBindAddress = serverBindAddress;

        // initialize all the default BrowserUp filter factories that provide core BUP functionality
        addBrowserUpFilters();

        HttpProxyServerBootstrap bootstrap = DefaultHttpProxyServer.bootstrap()
                .withFiltersSource(new HttpFiltersSource() {
                    @Override
                    public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext channelHandlerContext) {
                        return new BrowserUpHttpFilterChain(BrowserUpProxyServer.this, originalRequest, channelHandlerContext);
                    }

                    @Override
                    public int getMaximumRequestBufferSizeInBytes() {
                        return getMaximumRequestBufferSize();
                    }

                    @Override
                    public int getMaximumResponseBufferSizeInBytes() {
                        return getMaximumResponseBufferSize();
                    }
                })
                .withServerResolver(delegatingResolver)
                .withAddress(clientBindSocket)
                .withConnectTimeout(connectTimeoutMs)
                .withIdleConnectionTimeout(idleConnectionTimeoutSec)
                .withProxyAlias(VIA_HEADER_ALIAS);

        if (serverBindAddress != null) {
            bootstrap.withNetworkInterface(new InetSocketAddress(serverBindAddress, 0));
        }


        if (!mitmDisabled) {
            if (mitmManager == null) {
                mitmManager = ImpersonatingMitmManager.builder()
                        .rootCertificateSource(new KeyStoreFileCertificateSource(
                                KEYSTORE_TYPE,
                                useEcc ? EC_KEYSTORE_RESOURCE : RSA_KEYSTORE_RESOURCE,
                                KEYSTORE_PRIVATE_KEY_ALIAS,
                                KEYSTORE_PASSWORD))
                        .serverKeyGenerator(useEcc ? new ECKeyGenerator() : new RSAKeyGenerator())
                        .trustSource(trustSource)
                        .build();
            }

            bootstrap.withManInTheMiddle(mitmManager);
        }

        if (readBandwidthLimitBps > 0 || writeBandwidthLimitBps > 0) {
            bootstrap.withThrottling(readBandwidthLimitBps, writeBandwidthLimitBps);
        }

        if (chainedProxyManager != null) {
            bootstrap.withChainProxyManager(chainedProxyManager);
        } else if (upstreamProxyAddress != null) {
            // indicate that the proxy was bootstrapped with the default chained proxy manager, which allows changing the
            // chained proxy after the proxy is started.
            bootstrappedWithDefaultChainedProxy.set(true);

            bootstrap.withChainProxyManager((httpRequest, chainedProxies, clientDetails) -> {
                final InetSocketAddress upstreamProxy = upstreamProxyAddress;

                if (upstreamProxy != null) {

                    final boolean useEncryption = upstreamProxyHTTPS;
                    final List nonProxyHosts = upstreamProxyNonProxyHosts;

                    // skip upstream proxy configuration because the host is defined as proxy exception / non proxy hosts
                    // therefore we need to cast it to an URL
                    URL url = null;
                    try {
                        url = new URL(httpRequest.uri());
                    } catch (MalformedURLException e) {
                        log.error("The requested URL is not valid.", e);
                    }

                    final URL finalUrl = url; // need for lambda expression.
                    // note that we have to transform the wildcard like *.example.com to a valid regex
                    if (nonProxyHosts != null && finalUrl != null && nonProxyHosts.stream().anyMatch(nph -> finalUrl.getHost().matches(nph.trim().replace("*", ".*?")))) {
                        chainedProxies.add(ChainedProxyAdapter.FALLBACK_TO_DIRECT_CONNECTION);
                    } else {
                        chainedProxies.add(new ChainedProxyAdapter() {
                            @Override
                            public InetSocketAddress getChainedProxyAddress() {
                                return upstreamProxy;
                            }

                            @Override
                            public void filterRequest(HttpObject httpObject) {
                                String chainedProxyAuth = chainedProxyCredentials;
                                if (chainedProxyAuth != null) {
                                    if (httpObject instanceof HttpRequest) {
                                        if (ProxyUtils.isCONNECT(httpObject) || !((HttpRequest) httpObject).uri().startsWith("/")) {
                                            ((HttpRequest) httpObject).headers().add(HttpHeaderNames.PROXY_AUTHORIZATION, "Basic " + chainedProxyAuth);
                                        }
                                    }
                                }
                            }

                            @Override
                            public boolean requiresEncryption() {
                                return useEncryption;
                            }

                            @Override
                            public SSLEngine newSslEngine() {
                                if (useEncryption) {
                                    return new SelfSignedSslEngineSource(
                                            true, false).newSslEngine();
                                } else {
                                    return null;
                                }
                            }
                        });
                    }
                }
            });
        }

        if (threadPoolConfiguration != null) {
            bootstrap.withThreadPoolConfiguration(threadPoolConfiguration);
        }

        proxyServer = bootstrap.start();

        addHarCaptureFilter();

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                return new BlocklistFilter(originalRequest, ctx, getBlocklist());
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                Allowlist currentAllowlist = allowlist.get();
                return new AllowlistFilter(originalRequest, ctx, isAllowlistEnabled(), currentAllowlist.getStatusCode(), currentAllowlist.getPatterns());
            }
        });
    }

    @Override
    public boolean isStarted() {
        return started.get();
    }

    @Override
    public void start(int port) {
        this.start(port, null, null);
    }

    @Override
    public void start(int port, InetAddress bindAddress) {
        this.start(port, bindAddress, null);

    }

    @Override
    public void start() {
        this.start(0);
    }

    @Override
    public void stop() {
        stop(true);
    }

    @Override
    public void abort() {
        stop(false);
    }

    protected void stop(boolean graceful) {
        if (isStarted()) {
            if (stopped.compareAndSet(false, true)) {
                if (proxyServer != null) {
                    if (graceful) {
                        proxyServer.stop();
                    } else {
                        proxyServer.abort();
                    }
                } else {
                    log.warn("Attempted to stop proxy server, but proxy was never successfully started.");
                }
            } else {
                throw new IllegalStateException("Proxy server is already stopped. Cannot re-stop.");
            }
        } else {
            throw new IllegalStateException("Proxy server has not been started");
        }
    }

    @Override
    public InetAddress getClientBindAddress() {
        if (started.get()) {
            return proxyServer.getListenAddress().getAddress();
        } else {
            return null;
        }
    }

    @Override
    public int getPort() {
        if (started.get()) {
            return proxyServer.getListenAddress().getPort();
        } else {
            return 0;
        }
    }

    @Override
    public InetAddress getServerBindAddress() {
        return serverBindAddress;
    }

    @Override
    public Har getHar() {
        synchronized (GET_HAR_LOCK) {
            return this.har;
        }
    }

    @Override
    public Har getHar(boolean cleanHar) {
        if (!cleanHar) {
            return this.har;
        }

        synchronized (GET_HAR_LOCK) {
            return this.newHar();
        }
    }

    @Override
    public Har newHar() {
        return newHar(null);
    }

    @Override
    public Har newHar(String initialPageRef) {
        return newHar(initialPageRef, null);
    }

    @Override
    public Har newHar(String initialPageRef, String initialPageTitle) {
        return newHar(initialPageRef, initialPageTitle, true);
    }

    private Har newHar(String initialPageRef, String initialPageTitle, boolean createPage) {
        Har oldHar = endHar();

        addHarCaptureFilter();

        harPageCount.set(0);

        this.har = new Har();
        HarLog harLog = new HarLog();
        this.har.setLog(harLog);
        harLog.setCreator(HAR_CREATOR_VERSION);

        if (createPage) {
            newPage(initialPageRef, initialPageTitle);
        }

        return oldHar;
    }

    @Override
    public void setHarCaptureTypes(Set harCaptureSettings) {
        if (harCaptureSettings == null || harCaptureSettings.isEmpty()) {
            harCaptureTypes = EnumSet.noneOf(CaptureType.class);
        } else {
            harCaptureTypes = EnumSet.copyOf(harCaptureSettings);
        }
    }

    @Override
    public void setHarCaptureTypes(CaptureType... captureTypes) {
        if (captureTypes == null) {
            setHarCaptureTypes(EnumSet.noneOf(CaptureType.class));
        } else {
            setHarCaptureTypes(EnumSet.copyOf(Arrays.asList(captureTypes)));
        }
    }

    @Override
    public EnumSet getHarCaptureTypes() {
        return EnumSet.copyOf(harCaptureTypes);
    }

    @Override
    public void enableHarCaptureTypes(Set captureTypes) {
        harCaptureTypes.addAll(captureTypes);
    }

    @Override
    public void enableHarCaptureTypes(CaptureType... captureTypes) {
        if (captureTypes == null) {
            enableHarCaptureTypes(EnumSet.noneOf(CaptureType.class));
        } else {
            enableHarCaptureTypes(EnumSet.copyOf(Arrays.asList(captureTypes)));
        }
    }

    @Override
    public void disableHarCaptureTypes(Set captureTypes) {
        harCaptureTypes.removeAll(captureTypes);

    }

    @Override
    public void disableHarCaptureTypes(CaptureType... captureTypes) {
        if (captureTypes == null) {
            disableHarCaptureTypes(EnumSet.noneOf(CaptureType.class));
        } else {
            disableHarCaptureTypes(EnumSet.copyOf(Arrays.asList(captureTypes)));
        }
    }

    @Override
    public Har newPage() {
        return newPage(null);
    }

    @Override
    public Har newPage(String pageRef) {
        return newPage(pageRef, null);
    }

    @Override
    public Har newPage(String pageRef, String pageTitle) {
        har = getOrCreateHar(pageRef, pageTitle, false);

        Har endOfPageHar = null;

        if (currentHarPage != null) {
            String currentPageRef = currentHarPage.getId();

            // end the previous page, so that page-wide timings are populated
            endPage();

            // the interface requires newPage() to return the Har as it was immediately after the previous page was ended.
            endOfPageHar = BrowserUpProxyUtil.copyHarThroughPageRef(har, currentPageRef);
        }

        if (pageRef == null) {
            pageRef = "Page " + harPageCount.getAndIncrement();
        }

        if (pageTitle == null) {
            pageTitle = pageRef;
        }

        HarPage newPage = new HarPage();
        newPage.setTitle(pageTitle);
        newPage.setId(pageRef);
        newPage.setStartedDateTime(new Date());
        har.getLog().getPages().add(newPage);

        currentHarPage = newPage;

        return endOfPageHar;
    }

    @Override
    public Har endHar() {
        Har oldHar = getHar();
        if (oldHar == null) return null;

        // end the page and populate timings
        endPage();

        this.har = null;

        return oldHar;
    }

    @Override
    public void setReadBandwidthLimit(long bytesPerSecond) {
        this.readBandwidthLimitBps = bytesPerSecond;

        if (isStarted()) {
            proxyServer.setThrottle(this.readBandwidthLimitBps, this.writeBandwidthLimitBps);
        }
    }

    @Override
    public long getReadBandwidthLimit() {
        return readBandwidthLimitBps;
    }

    @Override
    public void setWriteBandwidthLimit(long bytesPerSecond) {
        this.writeBandwidthLimitBps = bytesPerSecond;

        if (isStarted()) {
            proxyServer.setThrottle(this.readBandwidthLimitBps, this.writeBandwidthLimitBps);
        }
    }

    private void updateDefaultPageTimings() {
        getDefaultPage().ifPresent(page -> {
            if (page.getStartedDateTime() != null) {
                page.getPageTimings().setOnLoad(Math.toIntExact(new Date().getTime() - page.getStartedDateTime().getTime()));
            }
        });
    }

    @Override
    public long getWriteBandwidthLimit() {
        return writeBandwidthLimitBps;
    }

    public void endPage() {
        if (har == null) return;

        HarPage previousPage = this.currentHarPage;
        this.currentHarPage = null;

        if (previousPage == null) {
            return;
        }

        if (previousPage.getStartedDateTime() != null) {
            previousPage
                    .getPageTimings()
                    .setOnLoad(Math.toIntExact(new Date().getTime() - previousPage.getStartedDateTime().getTime()));
        }

        updateDefaultPageTimings();
    }

    @Override
    public void addHeaders(Map headers) {
        ConcurrentMap newHeaders = new MapMaker().concurrencyLevel(1).makeMap();
        newHeaders.putAll(headers);

        this.additionalHeaders = newHeaders;
    }

    @Override
    public void setLatency(long latency, TimeUnit timeUnit) {
        this.latencyMs = (int) TimeUnit.MILLISECONDS.convert(latency, timeUnit);
    }

    @Override
    public void autoAuthorization(String domain, String username, String password, AuthType authType) {
        switch (authType) {
            case BASIC:
                // base64 encode the "username:password" string
                String base64EncodedCredentials = BrowserUpHttpUtil.base64EncodeBasicCredentials(username, password);

                basicAuthCredentials.put(domain, base64EncodedCredentials);
                break;

            default:
                throw new UnsupportedOperationException("AuthType " + authType + " is not supported for HTTP Authorization");
        }
    }

    @Override
    public void stopAutoAuthorization(String domain) {
        basicAuthCredentials.remove(domain);
    }

    @Override
    public void chainedProxyAuthorization(String username, String password, AuthType authType) {
        switch (authType) {
            case BASIC:
                chainedProxyCredentials = BrowserUpHttpUtil.base64EncodeBasicCredentials(username, password);
                break;

            default:
                throw new UnsupportedOperationException("AuthType " + authType + " is not supported for Proxy Authorization");
        }
    }

    @Override
    public void setConnectTimeout(int connectTimeout, TimeUnit timeUnit) {
        this.connectTimeoutMs = (int) TimeUnit.MILLISECONDS.convert(connectTimeout, timeUnit);

        if (isStarted()) {
            proxyServer.setConnectTimeout((int) TimeUnit.MILLISECONDS.convert(connectTimeout, timeUnit));
        }
    }

    /**
     * The LittleProxy implementation only allows idle connection timeouts to be specified in seconds. idleConnectionTimeouts greater than
     * 0 but less than 1 second will be set to 1 second; otherwise, values will be truncated (i.e. 1500ms will become 1s).
     */
    @Override
    public void setIdleConnectionTimeout(int idleConnectionTimeout, TimeUnit timeUnit) {
        long timeout = TimeUnit.SECONDS.convert(idleConnectionTimeout, timeUnit);
        if (timeout == 0 && idleConnectionTimeout > 0) {
            this.idleConnectionTimeoutSec = 1;
        } else {
            this.idleConnectionTimeoutSec = (int) timeout;
        }

        if (isStarted()) {
            proxyServer.setIdleConnectionTimeout(idleConnectionTimeoutSec);
        }
    }

    @Override
    public void setRequestTimeout(int requestTimeout, TimeUnit timeUnit) {
        //TODO: implement Request Timeouts using LittleProxy. currently this only sets an idle connection timeout, if the idle connection
        // timeout is higher than the specified requestTimeout.
        if (idleConnectionTimeoutSec == 0 || idleConnectionTimeoutSec > TimeUnit.SECONDS.convert(requestTimeout, timeUnit)) {
            setIdleConnectionTimeout(requestTimeout, timeUnit);
        }
    }

    @Override
    public void rewriteUrl(String pattern, String replace) {
        rewriteRules.add(new RewriteRule(pattern, replace));
    }

    @Override
    public void rewriteUrls(Map rewriteRules) {
        this.rewriteRules = rewriteRules.entrySet().stream()
                .map(rewriteRule -> new RewriteRule(rewriteRule.getKey(), rewriteRule.getValue()))
                .collect(toCollection(CopyOnWriteArrayList::new));
    }

    @Override
    public void clearRewriteRules() {
        rewriteRules.clear();
    }

    @Override
    public void blocklistRequests(String pattern, int responseCode) {
        blocklistEntries.add(new BlocklistEntry(pattern, responseCode));
    }

    @Override
    public void blocklistRequests(String pattern, int responseCode, String method) {
        blocklistEntries.add(new BlocklistEntry(pattern, responseCode, method));
    }

    @Override
    public void setBlocklist(Collection blocklist) {
        this.blocklistEntries = new CopyOnWriteArrayList<>(blocklist);
    }

    @Override
    public Collection getBlocklist() {
        return Collections.unmodifiableCollection(blocklistEntries);
    }

    @Override
    public boolean isAllowlistEnabled() {
        return allowlist.get().isEnabled();
    }

    @Override
    public Collection getAllowlistUrls() {
        ImmutableList.Builder builder = ImmutableList.builder();
        allowlist.get().getPatterns().stream()
                .map(Pattern::pattern)
                .forEach(builder::add);

        return builder.build();
    }

    @Override
    public int getAllowlistStatusCode() {
        return allowlist.get().getStatusCode();
    }

    @Override
    public void clearBlocklist() {
        blocklistEntries.clear();
    }

    @Override
    public void allowlistRequests(Collection urlPatterns, int statusCode) {
        this.allowlist.set(new Allowlist(urlPatterns, statusCode));
    }

    @Override
    public void addAllowlistPattern(String urlPattern) {
        // to make sure this method is threadsafe, we need to guarantee that the "snapshot" of the allowlist taken at the beginning
        // of the method has not been replaced by the time we have constructed a new allowlist at the end of the method
        boolean allowlistUpdated = false;
        while (!allowlistUpdated) {
            Allowlist currentAllowlist = this.allowlist.get();
            if (!currentAllowlist.isEnabled()) {
                throw new IllegalStateException("Allowlist is disabled. Cannot add patterns to a disabled allowlist.");
            }

            // retrieve the response code and list of patterns from the current allowlist, the construct a new list of patterns that contains
            // all of the old allowlist's patterns + this new pattern
            int statusCode = currentAllowlist.getStatusCode();
            List newPatterns = currentAllowlist.getPatterns().stream()
                    .map(Pattern::pattern)
                    .collect(toCollection(() -> new ArrayList<>(currentAllowlist.getPatterns().size() + 1)));
            newPatterns.add(urlPattern);

            // create a new (immutable) Allowlist object with the new pattern list and status code
            Allowlist newAllowlist = new Allowlist(newPatterns, statusCode);

            // replace the current allowlist with the new allowlist only if the current allowlist has not changed since we started
            allowlistUpdated = this.allowlist.compareAndSet(currentAllowlist, newAllowlist);
        }
    }

    /**
     * Allowlist the specified request patterns, returning the specified responseCode for non-allowlisted
     * requests.
     *
     * @param patterns     regular expression strings matching URL patterns to allowlist. if empty or null,
     *                     the allowlist will be enabled but will not match any URLs.
     * @param responseCode the HTTP response code to return for non-allowlisted requests
     */
    public void allowlistRequests(String[] patterns, int responseCode) {
        if (patterns == null || patterns.length == 0) {
            this.enableEmptyAllowlist(responseCode);
        } else {
            this.allowlistRequests(Arrays.asList(patterns), responseCode);
        }
    }

    @Override
    public void enableEmptyAllowlist(int statusCode) {
        allowlist.set(new Allowlist(statusCode));
    }

    @Override
    public void disableAllowlist() {
        allowlist.set(Allowlist.ALLOWLIST_DISABLED);
    }

    @Override
    public void addHeader(String name, String value) {
        additionalHeaders.put(name, value);
    }

    @Override
    public void removeHeader(String name) {
        additionalHeaders.remove(name);
    }

    @Override
    public void removeAllHeaders() {
        additionalHeaders.clear();
    }

    @Override
    public Map getAllHeaders() {
        return ImmutableMap.copyOf(additionalHeaders);
    }

    @Override
    public void setHostNameResolver(AdvancedHostResolver resolver) {
        delegatingResolver.setResolver(resolver);
    }

    @Override
    public AdvancedHostResolver getHostNameResolver() {
        return delegatingResolver.getResolver();
    }

    @Override
    public boolean waitForQuiescence(long quietPeriod, long timeout, TimeUnit timeUnit) {
        return activityMonitor.waitForQuiescence(quietPeriod, timeout, timeUnit);
    }

    /**
     * Instructs this proxy to route traffic through an upstream proxy.
     *
     * Note: Using {@link #setChainedProxyManager(ChainedProxyManager)} will supersede any value set by this method. A chained
     * proxy must be set before the proxy is started, though it can be changed after the proxy is started.
     *
     * @param chainedProxyAddress address of the upstream proxy
     */
    @Override
    public void setChainedProxy(InetSocketAddress chainedProxyAddress) {
        if (isStarted() && !bootstrappedWithDefaultChainedProxy.get()) {
            throw new IllegalStateException("Cannot set a chained proxy after the proxy is started if the proxy was started without a chained proxy.");
        }

        upstreamProxyAddress = chainedProxyAddress;
    }

    @Override
    public void setChainedProxyHTTPS(boolean chainedProxyHTTPS) {
        upstreamProxyHTTPS = chainedProxyHTTPS;
    }

    @Override
    public InetSocketAddress getChainedProxy() {
        return upstreamProxyAddress;
    }

    /**
     * Allows access to the LittleProxy {@link ChainedProxyManager} for fine-grained control of the chained proxies. To enable a single
     * chained proxy, {@link BrowserUpProxy#setChainedProxy(InetSocketAddress)} is generally more convenient.
     *
     * Note: The chained proxy manager must be enabled before calling {@link #start()}.
     *
     * @param chainedProxyManager chained proxy manager to enable
     */
    public void setChainedProxyManager(ChainedProxyManager chainedProxyManager) {
        if (isStarted()) {
            throw new IllegalStateException("Cannot configure chained proxy manager after proxy has started.");
        }

        this.chainedProxyManager = chainedProxyManager;
    }

    @Override
    public void setChainedProxyNonProxyHosts(List upstreamNonProxyHosts) {
        this.upstreamProxyNonProxyHosts = upstreamNonProxyHosts;
    }

    /**
     * Configures the Netty thread pool used by the LittleProxy back-end. See {@link ThreadPoolConfiguration} for details.
     *
     * @param threadPoolConfiguration thread pool configuration to use
     */
    public void setThreadPoolConfiguration(ThreadPoolConfiguration threadPoolConfiguration) {
        if (isStarted()) {
            throw new IllegalStateException("Cannot configure thread pool after proxy has started.");
        }

        this.threadPoolConfiguration = threadPoolConfiguration;
    }

    @Override
    public void addFirstHttpFilterFactory(HttpFiltersSource filterFactory) {
        filterFactories.add(0, filterFactory);
    }

    @Override
    public void addLastHttpFilterFactory(HttpFiltersSource filterFactory) {
        filterFactories.add(filterFactory);
    }

    /**
     * Note: The current implementation of this method forces a maximum response size of 2 MiB. To adjust the maximum response size, or
     * to disable aggregation (which disallows access to the {@link com.browserup.bup.util.HttpMessageContents}), you may add the filter source
     * directly: addFirstHttpFilterFactory(new ResponseFilterAdapter.FilterSource(filter, bufferSizeInBytes));
     */
    @Override
    public void addResponseFilter(ResponseFilter filter) {
        addLastHttpFilterFactory(new ResponseFilterAdapter.FilterSource(filter));
    }

    /**
     * Note: The current implementation of this method forces a maximum request size of 2 MiB. To adjust the maximum request size, or
     * to disable aggregation (which disallows access to the {@link com.browserup.bup.util.HttpMessageContents}), you may add the filter source
     * directly: addFirstHttpFilterFactory(new RequestFilterAdapter.FilterSource(filter, bufferSizeInBytes));
     */
    @Override
    public void addRequestFilter(RequestFilter filter) {
        addFirstHttpFilterFactory(new RequestFilterAdapter.FilterSource(filter));
    }

    @Override
    public Map getRewriteRules() {
        ImmutableMap.Builder builder = ImmutableMap.builder();
        rewriteRules.forEach(rewriteRule -> builder.put(rewriteRule.getPattern().pattern(), rewriteRule.getReplace()));

        return builder.build();
    }

    @Override
    public void removeRewriteRule(String urlPattern) {
        // normally removing elements from the list we are iterating over would not be possible, but since this is a CopyOnWriteArrayList
        // the iterator it returns is a "snapshot" of the list that will not be affected by removal (and that does not support removal, either)
        rewriteRules.stream()
                .filter(rewriteRule -> rewriteRule.getPattern().pattern().equals(urlPattern))
                .forEach(rewriteRule -> rewriteRules.remove(rewriteRule));
    }

    public boolean isStopped() {
        return stopped.get();
    }

    public HarPage getCurrentHarPage() {
        return currentHarPage;
    }

    public void addHttpFilterFactory(HttpFiltersSource filterFactory) {
        filterFactories.add(filterFactory);
    }

    public List getFilterFactories() {
        return filterFactories;
    }

    @Override
    public void setMitmDisabled(boolean mitmDisabled) throws IllegalStateException {
        if (isStarted()) {
            throw new IllegalStateException("Cannot disable MITM after the proxy has been started");
        }

        this.mitmDisabled = mitmDisabled;
    }

    @Override
    public void setMitmManager(MitmManager mitmManager) {
        this.mitmManager = mitmManager;
    }

    @Override
    public void setTrustAllServers(boolean trustAllServers) {
        if (isStarted()) {
            throw new IllegalStateException("Cannot disable upstream server verification after the proxy has been started");
        }

        if (trustAllServers) {
            trustSource = null;
        } else {
            if (trustSource == null) {
                trustSource = TrustSource.defaultTrustSource();
            }
        }
    }

    @Override
    public void setTrustSource(TrustSource trustSource) {
        if (isStarted()) {
            throw new IllegalStateException("Cannot change TrustSource after proxy has been started");
        }

        this.trustSource = trustSource;
    }

    @Override
    public Optional findMostRecentEntry(Pattern url) {
        List entries = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url).get();
        return Optional.ofNullable(entries.isEmpty() ? null : entries.get(0));
    }

    @Override
    public Collection findEntries(Pattern url) {
        return new UrlFilteredHarEntriesSupplier(getHar(), url).get();
    }

    @Override
    public AssertionResult assertMostRecentResponseTimeLessThanOrEqual(Pattern url, long time) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = new ResponseTimeLessThanOrEqualAssertion(time);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertResponseTimeLessThanOrEqual(Pattern url, long time) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new ResponseTimeLessThanOrEqualAssertion(time);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseContentContains(Pattern url, String text) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = new ContentContainsStringAssertion(text);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseContentDoesNotContain(Pattern url, String text) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = new ContentDoesNotContainStringAssertion(text);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseContentMatches(Pattern url, Pattern contentPattern) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = new ContentMatchesAssertion(contentPattern);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlContentLengthLessThanOrEquals(Pattern url, Long maxSize) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new ContentSizeLessThanOrEqualAssertion(maxSize);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlContentMatches(Pattern url, Pattern contentPattern) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new ContentMatchesAssertion(contentPattern);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlContentContains(Pattern url, String text) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new ContentContainsStringAssertion(text);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlContentDoesNotContain(Pattern url, String text) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new ContentDoesNotContainStringAssertion(text);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlResponseHeaderContains(Pattern url, String value) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new HeadersContainStringAssertion(value);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlResponseHeaderContains(Pattern url, String name, String value) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = StringUtils.isEmpty(name) ?
                new HeadersContainStringAssertion(value) :
                new FilteredHeadersContainStringAssertion(name, value);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlResponseHeaderDoesNotContain(Pattern url, String name, String value) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = StringUtils.isEmpty(name) ?
                new HeadersDoNotContainStringAssertion(value) :
                new FilteredHeadersDoNotContainStringAssertion(name, value);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlResponseHeaderDoesNotContain(Pattern url, String value) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new HeadersDoNotContainStringAssertion(value);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertAnyUrlResponseHeaderMatches(Pattern url, Pattern namePattern, Pattern valuePattern) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = namePattern == null ?
                new HeadersMatchAssertion(valuePattern) :
                new FilteredHeadersMatchAssertion(namePattern, valuePattern);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertResponseStatusCode(Integer status) {
        HarEntriesSupplier supplier = new CurrentStepHarEntriesSupplier(getHar());
        HarEntryAssertion assertion = new StatusEqualsAssertion(status);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertResponseStatusCode(HttpStatusClass clazz) {
        HarEntriesSupplier supplier = new CurrentStepHarEntriesSupplier(getHar());
        HarEntryAssertion assertion = new StatusBelongsToClassAssertion(clazz);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertResponseStatusCode(Pattern url, Integer status) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new StatusEqualsAssertion(status);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertResponseStatusCode(Pattern url, HttpStatusClass clazz) {
        HarEntriesSupplier supplier = new UrlFilteredHarEntriesSupplier(getHar(), url);
        HarEntryAssertion assertion = new StatusBelongsToClassAssertion(clazz);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseStatusCode(Integer status) {
        HarEntriesSupplier supplier = new MostRecentHarEntrySupplier(getHar());
        HarEntryAssertion assertion = new StatusEqualsAssertion(status);

        return checkAssertion(supplier, assertion);
    }


    @Override
    public AssertionResult assertMostRecentResponseStatusCode(HttpStatusClass clazz) {
        HarEntriesSupplier supplier = new MostRecentHarEntrySupplier(getHar());
        HarEntryAssertion assertion = new StatusBelongsToClassAssertion(clazz);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseStatusCode(Pattern url, Integer status) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = new StatusEqualsAssertion(status);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseStatusCode(Pattern url, HttpStatusClass clazz) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = new StatusBelongsToClassAssertion(clazz);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseContentLengthLessThanOrEqual(Pattern url, Long max) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = new ContentSizeLessThanOrEqualAssertion(max);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseHeaderContains(Pattern url, String name, String value) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = StringUtils.isEmpty(name) ?
                new HeadersContainStringAssertion(value) :
                new FilteredHeadersContainStringAssertion(name, value);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseHeaderDoesNotContain(Pattern url, String name, String value) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = StringUtils.isEmpty(name) ?
                new HeadersDoNotContainStringAssertion(value) :
                new FilteredHeadersDoNotContainStringAssertion(name, value);

        return checkAssertion(supplier, assertion);
    }

    @Override
    public AssertionResult assertMostRecentResponseHeaderMatches(Pattern url, Pattern name, Pattern value) {
        HarEntriesSupplier supplier = new MostRecentUrlFilteredHarEntrySupplier(getHar(), url);
        HarEntryAssertion assertion = name == null ?
                new HeadersMatchAssertion(value) :
                new FilteredHeadersMatchAssertion(name, value);

        return checkAssertion(supplier, assertion);
    }

    private AssertionResult checkAssertion(HarEntriesSupplier harEntriesSupplier, HarEntryAssertion assertion) {
        AssertionResult.Builder result = new AssertionResult.Builder();

        List entries = harEntriesSupplier.get();

        AtomicInteger failedCount = new AtomicInteger();

        entries.forEach(entry -> {
            AssertionEntryResult.Builder requestResult = new AssertionEntryResult.Builder();
            requestResult.setUrl(entry.getRequest().getUrl());

            Optional error = assertion.assertion(entry);
            requestResult.setFailed(error.isPresent());

            if (error.isPresent()) {
                requestResult.setMessage(error.get().getMessage());
                failedCount.getAndIncrement();
            }

            result.addRequest(requestResult.create());
        });

        String resultMessage = String.format("%d passed, %d total", entries.size() - failedCount.get(), entries.size());

        return result
                .setFilter(harEntriesSupplier.getFilterInfo())
                .setFailed(failedCount.get() > 0)
                .setMessage(resultMessage)
                .setPassed(failedCount.get() == 0)
                .create();
    }

    public boolean isMitmDisabled() {
        return this.mitmDisabled;
    }

    public void setUseEcc(boolean useEcc) {
        this.useEcc = useEcc;
    }

    /**
     * Adds the basic BrowserUp-proxy filters, except for the relatively-expensive HAR capture filter.
     */
    protected void addBrowserUpFilters() {
        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                return new ResolvedHostnameCacheFilter(originalRequest, ctx);
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                return new RegisterRequestFilter(originalRequest, ctx, activityMonitor);
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                return new HttpsOriginalHostCaptureFilter(originalRequest, ctx);
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                return new AutoBasicAuthFilter(originalRequest, ctx, basicAuthCredentials);
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                return new RewriteUrlFilter(originalRequest, ctx, rewriteRules);
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                return new HttpsHostCaptureFilter(originalRequest, ctx);
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest) {
                return new AddHeadersFilter(originalRequest, additionalHeaders);
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest) {
                return new LatencyFilter(originalRequest, latencyMs);
            }
        });

        addHttpFilterFactory(new HttpFiltersSourceAdapter() {
            @Override
            public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                return new UnregisterRequestFilter(originalRequest, ctx, activityMonitor);
            }
        });
    }

    private int getMaximumRequestBufferSize() {
        int maxBufferSize = 0;
        for (HttpFiltersSource source : filterFactories) {
            int requestBufferSize = source.getMaximumRequestBufferSizeInBytes();
            if (requestBufferSize > maxBufferSize) {
                maxBufferSize = requestBufferSize;
            }
        }

        return maxBufferSize;
    }

    private int getMaximumResponseBufferSize() {
        int maxBufferSize = 0;
        for (HttpFiltersSource source : filterFactories) {
            int requestBufferSize = source.getMaximumResponseBufferSizeInBytes();
            if (requestBufferSize > maxBufferSize) {
                maxBufferSize = requestBufferSize;
            }
        }

        return maxBufferSize;
    }

    /**
     * Enables the HAR capture filter if it has not already been enabled. The filter will be added to the end of the filter chain.
     * The HAR capture filter is relatively expensive, so this method is only called when a HAR is requested.
     */
    protected void addHarCaptureFilter() {
        if (harCaptureFilterEnabled.compareAndSet(false, true)) {
            // the HAR capture filter is (relatively) expensive, so only enable it when a HAR is being captured. furthermore,
            // restricting the HAR capture filter to requests where the HAR exists, as well as  excluding HTTP CONNECTs
            // from the HAR capture filter, greatly simplifies the filter code.
            addHttpFilterFactory(new HttpFiltersSourceAdapter() {
                @Override
                public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                    Har har = getOrCreateHar();
                    if (har != null && !ProxyUtils.isCONNECT(originalRequest)) {
                        return new HarCaptureFilter(originalRequest, ctx, har, getCurrentPageRef(), getHarCaptureTypes());
                    } else {
                        return null;
                    }
                }
            });

            // HTTP CONNECTs are a special case, since they require special timing and error handling
            addHttpFilterFactory(new HttpFiltersSourceAdapter() {
                @Override
                public HttpFilters filterRequest(HttpRequest originalRequest, ChannelHandlerContext ctx) {
                    Har har = getOrCreateHar();
                    if (har != null && ProxyUtils.isCONNECT(originalRequest)) {
                        return new HttpConnectHarCaptureFilter(originalRequest, ctx, har, getCurrentPageRef());
                    } else {
                        return null;
                    }
                }
            });
        }
    }

    private Har getOrCreateHar(String initialPageRef, String initialPageTitle, boolean createPage) {
        if (har == null) {
            synchronized (LOCK) {
                if (har == null) {
                    newHar(initialPageRef, initialPageTitle, createPage);
                }
            }
        }
        return har;
    }

    private Har getOrCreateHar() {
        return getOrCreateHar(DEFAULT_PAGE_REF, DEFAULT_PAGE_TITLE, true);
    }

    private String getCurrentPageRef() {
        HarPage harPage = getCurrentHarPage();
        harPage = harPage == null ? getOrCreateDefaultPage() : harPage;

        return harPage.getId();
    }

    private HarPage getOrCreateDefaultPage() {
        return getDefaultPage().orElseGet(this::addDefaultPage);
    }

    private HarPage addDefaultPage() {
        HarPage newPage = new HarPage();
        newPage.setTitle(DEFAULT_PAGE_REF);
        newPage.setStartedDateTime(new Date());
        newPage.setId(DEFAULT_PAGE_REF);
        getHar().getLog().getPages().add(newPage);
        return newPage;
    }

    private Optional getDefaultPage() {
        return getHar().getLog().getPages().stream()
                .filter(p -> p.getTitle().equals(DEFAULT_PAGE_REF))
                .findFirst();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy