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

liquibase.sdk.verifytest.TestPermutation Maven / Gradle / Ivy

package liquibase.sdk.verifytest;

import liquibase.exception.UnexpectedLiquibaseException;
import liquibase.sdk.exception.UnexpectedLiquibaseSdkException;
import liquibase.util.MD5Util;
import liquibase.util.StringUtils;

import java.util.*;

public class TestPermutation {

    private String notRanMessage;
    private SortedMap data = new TreeMap();
    private SortedMap description = new TreeMap();
    private String group;

    private TreeMap rowDescription;
    private String rowDescriptionParameter;
    private String rowFullKey = "";
    private String key = "";
    private String tableKey = "";
    private String fullKey = "";
    private SortedMap notes = new TreeMap();

    private List setupCommands = new ArrayList();
    private List verifications = new ArrayList();
    private List cleanupCommands = new ArrayList();

    private boolean valid = true;
    private boolean verified = false;
    private boolean canVerify;

    public static OkResult OK = new OkResult();

    public TestPermutation(VerifiedTest test) {
        test.addPermutation(this);
    }

    public String getKey() {
        return key;
    }

    public boolean getCanVerify() {
        return canVerify;
    }

    public void setCanVerify(boolean canVerify) {
        this.canVerify = canVerify;
    }

    public boolean isValid() {
        return valid;
    }

    public void setValid(boolean valid) {
        this.valid = valid;
    }

    public String getNotRanMessage() {
        return notRanMessage;
    }

    public void setNotRanMessage(String notRanMessage) {
        this.notRanMessage = notRanMessage;
    }

    public List getSetup() {
        return setupCommands;
    }

    public void addAssertion(Setup setup) {
        this.setupCommands.add(setup);
    }

    public void addSetup(Setup setup) {
        this.setupCommands.add(setup);
    }

    public SortedMap getDescription() {
        return description;
    }

    public String getGroup() {
        return group;
    }

    public String getRowDescriptionParameter() {
        return rowDescriptionParameter;
    }

    public Map getRowDescription() {
        return rowDescription;
    }

    public String getRowFullKey() {
        return rowFullKey;
    }

    public String getFullKey() {
        return fullKey;
    }

    public String getTableKey() {
        return tableKey;
    }

    public SortedMap getNotes() {
        return notes;
    }

    public SortedMap getData() {
        return data;
    }

    public void describe(String key, Object value) {
        describe(key, value, OutputFormat.DefaultFormat);
    }

    public void describe(String key, Object value, OutputFormat outputFormat) {
        description.put(key, new Value(value, outputFormat));
        recomputeKey();
    }

    public void describeAsGroup(String key, Object value) {
        describeAsGroup(key, value, OutputFormat.DefaultFormat);
    }

    public void describeAsGroup(String key, Object value, OutputFormat outputFormat) {
        this.group = key+": "+outputFormat.format(value);
        describe(key, value);
    }

    public void describeAsTable(String key, Map value) {
        describeAsTable(key, value, OutputFormat.DefaultFormat);
    }

    public void describeAsTable(String key, Map value, OutputFormat outputFormat) {
        rowDescriptionParameter = key;
        rowDescription = new TreeMap();
        for (Map.Entry entry : value.entrySet()) {
            rowDescription.put(entry.getKey(), new Value(entry.getValue(), outputFormat));
        }

        recomputeKey();
    }

    protected void recomputeKey() {
        SortedMap fullDescription = new TreeMap(description);
        StringUtils.StringUtilsFormatter formatter = new StringUtils.StringUtilsFormatter() {
            @Override
            public String toString(Object obj) {
                return ((Value) obj).serialize();
            }
        };

        if (rowDescription != null) {
            for (Map.Entry rowEntry : rowDescription.entrySet()) {
                fullDescription.put(rowEntry.getKey(), rowEntry.getValue());
            }
            rowFullKey = StringUtils.join(rowDescription, ",", formatter);
        }
        tableKey = StringUtils.join(description, ",", formatter);
        fullKey = StringUtils.join(fullDescription, ",", formatter);
        key = MD5Util.computeMD5(fullKey).substring(0, 16);
    }

    public void note(String key, Object value) {
        note(key, value, OutputFormat.DefaultFormat);
    }

    public void note(String key, Object value, OutputFormat outputFormat) {
        notes.put(key, new Value(value, outputFormat));
    }

    public void data(String key, Object value) {
        data(key, value, OutputFormat.DefaultFormat);
    }

    public void data(String key, Object value, OutputFormat outputFormat) {
        data.put(key, new Value(value, outputFormat));
    }

    public List getVerifications() {
        return verifications;
    }

    public void addVerification(Verification verification) {
        verifications.add(verification);
    }

    public List getCleanup() {
        return cleanupCommands;
    }

    public void addCleanup(Cleanup cleanup) {
        cleanupCommands.add(cleanup);
    }

    public void test(VerifiedTest test) throws Exception {
        TestPermutation previousRun = VerifiedTestFactory.getInstance().getSavedRun(test, this);

        if (notRanMessage != null) {
            save(test);
            return;
        }

        try {
            for (Setup setup : this.setupCommands) {
                SetupResult result = setup.run();

                if (result == null) {
                    throw new UnexpectedLiquibaseException("No result returned by setup");
                } else {
                    if (!result.isValid()) {
                        valid = false;
                        canVerify = false;
                        notRanMessage = result.getMessage();
                        break;
                    } else if (!result.canVerify()) {
                        canVerify = false;
                        notRanMessage = result.getMessage();
                    }
                }
            }
        } catch (Throwable e) {
            SortedMap fullDescription = new TreeMap(description);
            if (rowDescription != null) {
                fullDescription.putAll(rowDescription);
            }
            String message = "Error executing setup\n"+
                    "Description: "+ output(fullDescription)+"\n"+
                    "Notes: "+output(notes)+"\n"+
                    "Data: "+output(data);
            throw new RuntimeException(message, e);
        }

        if (!valid || !canVerify) {
            save(test);
            return;
        }

        Exception cleanupError = null;
        try {
            try {
                for (Verification verification : this.verifications) {
                    verification.run();
                }
            } catch (CannotVerifyException e) {
                this.verified = false;
            } catch (Throwable e) {
                String message = "Error executing verification\n"+
                        "Description: "+ output(description)+"\n"+
                        (rowDescription == null ? "" : "Row Description: "+output(rowDescription)+"\n")+
                        "Notes: "+output(notes)+"\n"+
                        "Data: "+output(data);
                throw new RuntimeException(message, e);
            }
            this.verified = true;
        } finally {
            for (Cleanup cleanup : cleanupCommands) {
                try {
                    cleanup.run();
                } catch (Exception e) {
                    cleanupError = e;
                }
            }
        }

        if (cleanupError != null) {
            throw new UnexpectedLiquibaseSdkException("Cleanup error", cleanupError);
        }

        save(test);
    }

    protected void save(VerifiedTest test) throws Exception {
        VerifiedTestFactory.getInstance().saveRun(test, this);
    }

    private String output(SortedMap map) {
        List out = new ArrayList();
        for (Map.Entry entry : map.entrySet()) {
            out.add(entry.getKey()+"="+entry.getValue().serialize());
        }

        return StringUtils.join(out, ", ");
    }

    public boolean getVerified() {
        return verified;
    }

    public void setVerified(boolean verified) {
        this.verified = verified;
    }

    public static interface SetupResult {
        boolean isValid();
        boolean canVerify();
        String getMessage();
    }

    public static class Invalid implements SetupResult {

        private String message;

        public Invalid(String message) {
            this.message = message;
        }

        @Override
        public boolean isValid() {
            return false;
        }

        @Override
        public boolean canVerify() {
            return false;
        }

        @Override
        public String getMessage() {
            return message;
        }
    }

    public static class CannotVerify implements SetupResult {

        private String message;

        public CannotVerify(String message) {
            this.message = message;
        }

        @Override
        public boolean isValid() {
            return true;
        }

        @Override
        public boolean canVerify() {
            return false;
        }

        @Override
        public String getMessage() {
            return message;
        }
    }


    public static class OkResult implements SetupResult {

        public OkResult() {
        }

        @Override
        public boolean isValid() {
            return true;
        }

        @Override
        public boolean canVerify() {
            return true;
        }

        @Override
        public String getMessage() {
            return null;
        }
    }



    public static interface Setup {
        public SetupResult run();
    }

    public static interface Verification {
        public void run();
    }

    public static interface Cleanup {
        public void run();
    }

    public static class CannotVerifyException extends RuntimeException {
        public CannotVerifyException(String message) {
            super(message);
        }

        public CannotVerifyException(String message, Throwable cause) {
            super(message, cause);
        }
    }


    public static class Value {
        private Object value;
        private OutputFormat format;
        private String stringValue;

        public Value(Object value, OutputFormat format) {
            this.value = value;
            this.format = format;
        }

        public Object getValue() {
            return value;
        }

        public String serialize() {
            if (stringValue == null) {
                stringValue = format.format(value);
            }
            return stringValue;
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy