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

io.micronaut.http.tck.BodyAssertion Maven / Gradle / Ivy

/*
 * Copyright 2017-2022 original authors
 *
 * 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
 *
 * https://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 io.micronaut.http.tck;

import io.micronaut.core.annotation.Experimental;
import io.micronaut.core.annotation.Nullable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.Arrays;
import java.util.function.BiPredicate;
import java.util.stream.Collectors;
import java.util.stream.IntStream;

import static org.junit.jupiter.api.Assertions.assertTrue;

/**
 * HTTP Response's body assertions.
 *
 * @param  The body type
 * @param  The error type
 */
@Experimental
public final class BodyAssertion {
    private static final Logger LOG = LoggerFactory.getLogger(BodyAssertion.class);

    private final Class bodyType;
    private final Class errorType;
    private final T expected;
    private final BodyEvaluator evaluator;

    private BodyAssertion(Class bodyType,
                          Class errorType,
                          T expected,
                          BodyEvaluator evaluator
    ) {
        this.bodyType = bodyType;
        this.errorType = errorType;
        this.expected = expected;
        this.evaluator = evaluator;
    }

    /**
     * @return a Builder;
     */
    public static BodyAssertion.Builder builder() {
        return new BodyAssertion.Builder();
    }

    /**
     * Evaluates the HTTP Response Body.
     *
     * @param body The HTTP Response Body
     */
    @SuppressWarnings("java:S5960") // Assertion is the whole point of this method
    public void evaluate(T body) {
        assertTrue(this.evaluator.test(expected, body), () -> this.evaluator.message(expected, body));
    }

    /**
     * @return The expected body type
     */
    public Class getBodyType() {
        return bodyType;
    }

    @Nullable
    public Class getErrorType() {
        return errorType;
    }

    private enum EvaluatorType {
        EQUAL,
        CONTAIN,
    }

    /**
     * The interface for typed BodyAssertion Builders.
     *
     * @param  The body type
     * @param  The error type
     */
    public interface AssertionBuilder {

        /**
         * @return a body assertion which verifiers the HTTP Response's body contains the expected body
         */
        BodyAssertion contains();

        /**
         * @return a body assertion which verifiers the HTTP Response's body is equals to the expected body
         */
        BodyAssertion equals();
    }

    private interface BodyEvaluator extends BiPredicate {

        EvaluatorType type();

        default String render(T value) {
            return String.valueOf(value);
        }

        default String message(T expected, T actual) {
            return "Expected received body of '" + render(actual) + "' to " + type().name().toLowerCase() + " '" + render(expected) + "'";
        }
    }

    /**
     * BodyAssertion Builder.
     */
    public static class Builder {

        /**
         * @param expected Expected Body
         * @return The Builder
         */
        public AssertionBuilder body(String expected) {
            return new StringBodyAssertionBuilder(expected);
        }

        /**
         * @param expected Expected Body
         * @return The Builder
         */
        public AssertionBuilder body(byte[] expected) {
            return new ByteArrayBodyAssertionBuilder(expected);
        }
    }

    /**
     * String BodyAssertion Builder.
     */
    public static class StringBodyAssertionBuilder extends BodyAssertion.Builder implements AssertionBuilder {

        private final String body;

        public StringBodyAssertionBuilder(String expected) {
            this.body = expected;
        }

        /**
         * @return a body assertion which verifiers the HTTP Response's body contains the expected body
         */
        @Override
        public BodyAssertion contains() {
            return new BodyAssertion<>(String.class, String.class, this.body, new StringEvaluator(EvaluatorType.CONTAIN));
        }

        /**
         * @return a body assertion which verifiers the HTTP Response's body is equals to the expected body
         */
        @Override
        public BodyAssertion equals() {
            return new BodyAssertion<>(String.class, String.class, this.body, new StringEvaluator(EvaluatorType.EQUAL));
        }
    }

    /**
     * Byte Array BodyAssertion Builder.
     */
    public static class ByteArrayBodyAssertionBuilder extends BodyAssertion.Builder implements BodyAssertion.AssertionBuilder {

        private final byte[] body;

        public ByteArrayBodyAssertionBuilder(byte[] expected) {
            this.body = expected;
        }

        /**
         * @return a body assertion which verifiers the HTTP Response's body contains the expected body
         */
        @Override
        public BodyAssertion contains() {
            return new BodyAssertion<>(byte[].class, byte[].class, this.body, new ByteArrayEvaluator(EvaluatorType.CONTAIN));
        }

        /**
         * @return a body assertion which verifiers the HTTP Response's body is equals to the expected body
         */
        @Override
        public BodyAssertion equals() {
            return new BodyAssertion<>(byte[].class, byte[].class, this.body, new ByteArrayEvaluator(EvaluatorType.EQUAL));
        }
    }

    private record StringEvaluator(EvaluatorType type) implements BodyEvaluator {

        @Override
        public boolean test(String expected, String received) {
            return switch (type) {
                case EQUAL -> received.equals(expected);
                case CONTAIN -> received.contains(expected);
            };
        }
    }

    private record ByteArrayEvaluator(EvaluatorType type) implements BodyEvaluator {

        @Override
        public String render(byte[] value) {
            if (value == null) {
                return "null";
            }
            String firstTen = IntStream.range(0, value.length)
                .map(i -> value[i] & 0xff)
                .mapToObj(i -> "%02x".formatted(i))
                .limit(10)
                .collect(Collectors.joining(", ", "", "..."));
            return "ByteArray(length=" + value.length + ", [" + firstTen + "])";
        }

        @Override
        public boolean test(byte[] expected, byte[] received) {
            return switch (type) {
                case EQUAL -> Arrays.equals(received, expected);
                case CONTAIN -> throw new AssertionError("Not implemented yet!");
            };
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy