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

io.qameta.allure.Allure Maven / Gradle / Ivy

There is a newer version: 2.29.1
Show newest version
/*
 *  Copyright 2016-2024 Qameta Software Inc
 *
 *  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 io.qameta.allure;

import io.qameta.allure.model.Label;
import io.qameta.allure.model.Link;
import io.qameta.allure.model.Parameter;
import io.qameta.allure.model.Status;
import io.qameta.allure.model.StepResult;
import io.qameta.allure.util.ExceptionUtils;

import java.io.ByteArrayInputStream;
import java.io.InputStream;
import java.nio.charset.StandardCharsets;
import java.util.Objects;
import java.util.UUID;
import java.util.concurrent.CompletableFuture;
import java.util.function.Supplier;

import static io.qameta.allure.util.ResultsUtils.EPIC_LABEL_NAME;
import static io.qameta.allure.util.ResultsUtils.FEATURE_LABEL_NAME;
import static io.qameta.allure.util.ResultsUtils.ISSUE_LINK_TYPE;
import static io.qameta.allure.util.ResultsUtils.STORY_LABEL_NAME;
import static io.qameta.allure.util.ResultsUtils.SUITE_LABEL_NAME;
import static io.qameta.allure.util.ResultsUtils.TMS_LINK_TYPE;
import static io.qameta.allure.util.ResultsUtils.createParameter;
import static io.qameta.allure.util.ResultsUtils.getStatus;
import static io.qameta.allure.util.ResultsUtils.getStatusDetails;
import static java.util.Arrays.asList;
import static java.util.concurrent.CompletableFuture.supplyAsync;

/**
 * The class contains some useful methods to work with {@link AllureLifecycle}.
 */
@SuppressWarnings({"PMD.ClassNamingConventions", "PMD.ExcessivePublicCount", "PMD.TooManyMethods"})
public final class Allure {

    private static final String TXT_EXTENSION = ".txt";
    private static final String TEXT_PLAIN = "text/plain";

    private static AllureLifecycle lifecycle;

    /**
     * Do not instance.
     */
    private Allure() {
        throw new IllegalStateException("Do not instance");
    }

    /**
     * Returns {@link AllureLifecycle} for low level operations with results.
     *
     * @return the lifecycle.
     */
    public static AllureLifecycle getLifecycle() {
        if (Objects.isNull(lifecycle)) {
            lifecycle = new AllureLifecycle();
        }
        return lifecycle;
    }

    /**
     * Sets {@link AllureLifecycle}.
     */
    public static void setLifecycle(final AllureLifecycle lifecycle) {
        Allure.lifecycle = lifecycle;
    }

    /**
     * Adds passed step with provided name in current test or step (or test fixture). Takes no effect
     * if no test run at the moment. Shortcut for {@link #step(String, Status)}.
     *
     * @param name the name of step.
     */
    public static void step(final String name) {
        step(name, Status.PASSED);
    }

    /**
     * Adds step with provided name and status in current test or step (or test fixture). Takes no effect
     * if no test run at the moment.
     *
     * @param name   the name of step.
     * @param status the step status.
     */
    public static void step(final String name, final Status status) {
        final String uuid = UUID.randomUUID().toString();
        getLifecycle().startStep(uuid, new StepResult().setName(name).setStatus(status));
        getLifecycle().stopStep(uuid);
    }

    /**
     * Syntax sugar for {@link #step(String, ThrowableRunnable)}.
     *
     * @param name     the name of step.
     * @param runnable the step's body.
     */
    public static void step(final String name, final ThrowableRunnableVoid runnable) {
        step(name, () -> {
            runnable.run();
            return null;
        });
    }

    /**
     * Syntax sugar for {@link #step(ThrowableContextRunnable)}.
     *
     * @param name     the name of step.
     * @param runnable the step's body.
     */
    public static  T step(final String name, final ThrowableRunnable runnable) {
        return step(step -> {
            step.name(name);
            return runnable.run();
        });
    }

    /**
     * Syntax sugar for {@link #step(ThrowableContextRunnable)}.
     *
     * @param runnable the step's body.
     */
    public static void step(final ThrowableContextRunnableVoid runnable) {
        step(step -> {
            runnable.run(step);
            return null;
        });
    }

    /**
     * Syntax sugar for {@link #step(ThrowableContextRunnable)}.
     *
     * @param name     the name of step.
     * @param runnable the step's body.
     */
    public static void step(final String name, final ThrowableContextRunnableVoid runnable) {
        step(step -> {
            step.name(name);
            runnable.run(step);
            return null;
        });
    }

    /**
     * Syntax sugar for {@link #step(ThrowableContextRunnable)}.
     *
     * @param name     the name of step.
     * @param runnable the step's body.
     */
    public static  T step(final String name, final ThrowableContextRunnable runnable) {
        return step(step -> {
            step.name(name);
            return runnable.run(step);
        });
    }

    /**
     * Run provided {@link ThrowableRunnable} as step with given name. Takes no effect
     * if no test run at the moment.
     *
     * @param runnable the step's body.
     */
    public static  T step(final ThrowableContextRunnable runnable) {
        final String uuid = UUID.randomUUID().toString();
        getLifecycle().startStep(uuid, new StepResult().setName("step"));

        try {
            final T result = runnable.run(new DefaultStepContext(uuid));
            getLifecycle().updateStep(uuid, step -> step.setStatus(Status.PASSED));
            return result;
        } catch (Throwable throwable) {
            getLifecycle().updateStep(s -> s
                    .setStatus(getStatus(throwable).orElse(Status.BROKEN))
                    .setStatusDetails(getStatusDetails(throwable).orElse(null)));
            throw ExceptionUtils.sneakyThrow(throwable);
        } finally {
            getLifecycle().stopStep(uuid);
        }
    }

    /**
     * Adds epic label to current test if any. Takes no effect
     * if no test run at the moment. Shortcut for {@link #label(String, String)}.
     *
     * @param value the value of label.
     */
    public static void epic(final String value) {
        label(EPIC_LABEL_NAME, value);
    }

    /**
     * Adds feature label to current test if any. Takes no effect
     * if no test run at the moment. Shortcut for {@link #label(String, String)}.
     *
     * @param value the value of label.
     */
    public static void feature(final String value) {
        label(FEATURE_LABEL_NAME, value);
    }

    /**
     * Adds story label to current test if any. Takes no effect
     * if no test run at the moment. Shortcut for {@link #label(String, String)}.
     *
     * @param value the value of label.
     */
    public static void story(final String value) {
        label(STORY_LABEL_NAME, value);
    }

    /**
     * Adds suite label to current test if any. Takes no effect
     * if no test run at the moment. Shortcut for {@link #label(String, String)}.
     *
     * @param value the value of label.
     */
    public static void suite(final String value) {
        label(SUITE_LABEL_NAME, value);
    }

    /**
     * Adds label to current test if any. Takes no effect
     * if no test run at the moment.
     *
     * @param name  the name of label.
     * @param value the value of label.
     */
    public static void label(final String name, final String value) {
        final Label label = new Label().setName(name).setValue(value);
        getLifecycle().updateTestCase(testResult -> testResult.getLabels().add(label));
    }

    /**
     * Adds parameter to current test if any. Takes no effect
     * if no test run at the moment.
     * 

* Shortcut for {@link #parameter(String, Object, Boolean, Parameter.Mode)}. * * @param name the name of parameter. * @param value the value of parameter. */ public static T parameter(final String name, final T value) { return parameter(name, value, null, null); } /** * Adds parameter to current test if any. Takes no effect * if no test run at the moment. *

* Shortcut for {@link #parameter(String, Object, Boolean, Parameter.Mode)}. * * @param name the name of parameter. * @param value the value of parameter. * @param excluded true if parameter should be excluded from history key calculation, false otherwise. * @return the specified value. */ public static T parameter(final String name, final T value, final Boolean excluded) { return parameter(name, value, excluded, null); } /** * Adds parameter to current test if any. Takes no effect * if no test run at the moment. *

* Shortcut for {@link #parameter(String, Object, Boolean, Parameter.Mode)}. * * @param name the name of parameter. * @param value the value of parameter. * @param mode the parameter mode. * @return the specified value. */ public static T parameter(final String name, final T value, final Parameter.Mode mode) { return parameter(name, value, null, mode); } /** * Adds parameter to current test if any. Takes no effect * if no test run at the moment. * * @param name the name of parameter. * @param value the value of parameter. * @param excluded true if parameter should be excluded from history key calculation, false otherwise. * @param mode the parameter mode. * @return the specified value. */ public static T parameter(final String name, final T value, final Boolean excluded, final Parameter.Mode mode) { final Parameter parameter = createParameter(name, value, excluded, mode); getLifecycle().updateTestCase(testResult -> testResult.getParameters().add(parameter)); return value; } /** * Adds issue link to current test if any. Takes no effect * if no test run at the moment. Shortcut for {@link #link(String, String, String)}. * * @param name the name of link. * @param url the link's url. */ public static void issue(final String name, final String url) { link(name, ISSUE_LINK_TYPE, url); } /** * Adds tms link to current test if any. Takes no effect * if no test run at the moment. Shortcut for {@link #link(String, String, String)}. * * @param name the name of link. * @param url the link's url. */ public static void tms(final String name, final String url) { link(name, TMS_LINK_TYPE, url); } /** * Adds link to current test if any. Takes no effect * if no test run at the moment. Shortcut for {@link #link(String, String)} * * @param url the link's url. */ public static void link(final String url) { link(null, url); } /** * Adds link to current test if any. Takes no effect * if no test run at the moment. Shortcut for {@link #link(String, String, String)} * * @param name the name of link. * @param url the link's url. */ public static void link(final String name, final String url) { link(name, null, url); } /** * Adds link to current test if any. Takes no effect * if no test run at the moment. * * @param name the name of link. * @param type the type of link, used to display link icon in the report. * @param url the link's url. */ public static void link(final String name, final String type, final String url) { final Link link = new Link().setName(name).setType(type).setUrl(url); getLifecycle().updateTestCase(testResult -> testResult.getLinks().add(link)); } /** * Adds description to current test if any. Takes no effect * if no test run at the moment. Expecting description provided in Markdown format. * * @param description the description in markdown format. * @see #descriptionHtml(String) */ public static void description(final String description) { getLifecycle().updateTestCase(executable -> executable.setDescription(description)); } /** * Adds descriptionHtml to current test if any. Takes no effect * if no test run at the moment. Note that description will take no effect if descriptionHtml is * specified. * * @param descriptionHtml the description in html format. * @see #description(String) */ public static void descriptionHtml(final String descriptionHtml) { getLifecycle().updateTestCase(executable -> executable.setDescriptionHtml(descriptionHtml)); } /** * Adds attachment. * * @param name the name of attachment. * @param content the attachment content. */ public static void attachment(final String name, final String content) { addAttachment(name, content); } /** * Adds attachment. * * @param name the name of attachment. * @param content the stream that contains attachment content. */ public static void attachment(final String name, final InputStream content) { addAttachment(name, content); } /** * @deprecated use {@link #label(String, String)} instead. */ @Deprecated public static void addLabels(final Label... labels) { getLifecycle().updateTestCase(testResult -> testResult.getLabels().addAll(asList(labels))); } /** * @deprecated use {@link #link(String, String, String)} instead. */ @Deprecated public static void addLinks(final Link... links) { getLifecycle().updateTestCase(testResult -> testResult.getLinks().addAll(asList(links))); } /** * @deprecated use {@link #description(String)} instead. */ @Deprecated public static void addDescription(final String description) { description(description); } /** * @deprecated use {@link #descriptionHtml(String)} instead. */ @Deprecated public static void addDescriptionHtml(final String descriptionHtml) { descriptionHtml(descriptionHtml); } public static void addAttachment(final String name, final String content) { getLifecycle().addAttachment(name, TEXT_PLAIN, TXT_EXTENSION, content.getBytes(StandardCharsets.UTF_8)); } public static void addAttachment(final String name, final String type, final String content) { getLifecycle().addAttachment(name, type, TXT_EXTENSION, content.getBytes(StandardCharsets.UTF_8)); } @SuppressWarnings("PMD.UseObjectForClearerAPI") public static void addAttachment(final String name, final String type, final String content, final String fileExtension) { getLifecycle().addAttachment(name, type, fileExtension, content.getBytes(StandardCharsets.UTF_8)); } public static void addAttachment(final String name, final InputStream content) { getLifecycle().addAttachment(name, null, null, content); } @SuppressWarnings("PMD.UseObjectForClearerAPI") public static void addAttachment(final String name, final String type, final InputStream content, final String fileExtension) { getLifecycle().addAttachment(name, type, fileExtension, content); } public static CompletableFuture addByteAttachmentAsync( final String name, final String type, final Supplier body) { return addByteAttachmentAsync(name, type, "", body); } public static CompletableFuture addByteAttachmentAsync( final String name, final String type, final String fileExtension, final Supplier body) { final String source = getLifecycle().prepareAttachment(name, type, fileExtension); return supplyAsync(body).whenComplete((result, ex) -> getLifecycle().writeAttachment(source, new ByteArrayInputStream(result))); } public static CompletableFuture addStreamAttachmentAsync( final String name, final String type, final Supplier body) { return addStreamAttachmentAsync(name, type, "", body); } public static CompletableFuture addStreamAttachmentAsync( final String name, final String type, final String fileExtension, final Supplier body) { final String source = lifecycle.prepareAttachment(name, type, fileExtension); return supplyAsync(body).whenComplete((result, ex) -> lifecycle.writeAttachment(source, result)); } /** * Runnable that allows to throw an exception and return any type. * * @param the type of return value. */ @FunctionalInterface public interface ThrowableRunnable { T run() throws Throwable; } /** * Runnable that allows to throw an exception and return void. */ @FunctionalInterface public interface ThrowableRunnableVoid { void run() throws Throwable; } /** * Runnable that allows to throw an exception and return any type. * * @param the type of return value. * @param the type of context. */ @FunctionalInterface public interface ThrowableContextRunnable { T run(U context) throws Throwable; } /** * Callable that allows to throw an exception and return void. * * @param the type of context. */ @FunctionalInterface public interface ThrowableContextRunnableVoid { void run(T context) throws Throwable; } /** * Step context. */ @SuppressWarnings("MultipleStringLiterals") public interface StepContext { /** * Sets step's name. * * @param name the deserted name of step. */ void name(String name); /** * Adds parameter to a step. * * @param name the name of parameter. * @param value the value. * @param the type of value. * @return the value. */ T parameter(String name, T value); /** * Adds parameter to a step. * * @param name the name of parameter. * @param value the value. * @param excluded true if parameter should be excluded from history key generation, false otherwise. * @param the type of value. * @return the value. */ default T parameter(final String name, final T value, final Boolean excluded) { throw new UnsupportedOperationException("method is not implemented"); } /** * Adds parameter to a step. * * @param name the name of parameter. * @param value the value. * @param mode the parameter's mode. * @param the type of value. * @return the value. */ default T parameter(final String name, final T value, final Parameter.Mode mode) { throw new UnsupportedOperationException("method is not implemented"); } /** * Adds parameter to a step. * * @param name the name of parameter. * @param value the value. * @param excluded true if parameter should be excluded from history key generation, false otherwise. * @param mode the parameter's mode. * @param the type of value. * @return the value. */ default T parameter(final String name, final T value, final Boolean excluded, final Parameter.Mode mode) { throw new UnsupportedOperationException("method is not implemented"); } } /** * Basic implementation of step context. */ private static final class DefaultStepContext implements StepContext { private final String uuid; private DefaultStepContext(final String uuid) { this.uuid = uuid; } @Override public void name(final String name) { getLifecycle().updateStep(uuid, stepResult -> stepResult.setName(name)); } @Override public T parameter(final String name, final T value) { return parameter(name, value, null, null); } @Override public T parameter(final String name, final T value, final Boolean excluded) { return parameter(name, value, excluded, null); } @Override public T parameter(final String name, final T value, final Parameter.Mode mode) { return parameter(name, value, null, mode); } @Override public T parameter(final String name, final T value, final Boolean excluded, final Parameter.Mode mode) { final Parameter param = createParameter(name, value, excluded, mode); getLifecycle().updateStep(uuid, stepResult -> stepResult.getParameters().add(param)); return value; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy