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

org.dmfs.jems2.hamcrest.matchers.charsequence.CharSequenceSubSequenceMatcher Maven / Gradle / Ivy

/*
 * Copyright 2021 dmfs GmbH
 *
 *
 * 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 org.dmfs.jems2.hamcrest.matchers.charsequence;

import org.dmfs.jems2.iterable.Seq;
import org.hamcrest.Description;
import org.hamcrest.Matcher;
import org.hamcrest.TypeSafeDiagnosingMatcher;
import org.hamcrest.core.AllOf;

import java.util.Locale;

import static org.dmfs.jems2.hamcrest.matchers.charsequence.CharSequenceCharAtMatcher.hasChars;
import static org.dmfs.jems2.hamcrest.matchers.charsequence.CharSequenceLengthMatcher.hasLength;
import static org.hamcrest.Matchers.hasToString;


/**
 * A {@link Matcher} to check that all sub-sequences of a {@link CharSequence} are valid {@link CharSequence}s themselves.
 */
public final class CharSequenceSubSequenceMatcher extends TypeSafeDiagnosingMatcher
{
    private final CharSequence mExpectedValue;
    private final int mSubSequenceTestDepth;


    /**
     * Test that sub-sequences of a {@link CharSequence} are valid {@link CharSequence}s as well.
     *
     * @param expectedValue
     *     The value to compare.
     * @param subSequenceTestDepth
     *     The recursion depth.
     *
     * @return A {@link CharSequenceSubSequenceMatcher}
     */
    public static CharSequenceSubSequenceMatcher hasSubSequences(CharSequence expectedValue, int subSequenceTestDepth)
    {
        return new CharSequenceSubSequenceMatcher(expectedValue, subSequenceTestDepth);
    }


    public CharSequenceSubSequenceMatcher(CharSequence expectedValue, int subSequenceTestDepth)
    {
        mExpectedValue = expectedValue;
        mSubSequenceTestDepth = subSequenceTestDepth;
    }


    @Override
    protected boolean matchesSafely(CharSequence item, Description mismatchDescription)
    {
        if (mSubSequenceTestDepth == 0)
        {
            // don't test any deeper to avoid infinite loops
            return true;
        }
        int max = mExpectedValue.length();

        // note, this loop can be quite intense for longer charSequences, consider reducing the tested range
        for (int i = -max - 1; i <= max * 2 + 1; ++i)
        {
            for (int j = -max - 1; j <= max * 2 + 1; ++j)
            {
                if (i < 0 || j > max || j < i)
                {
                    // test illegal indexes
                    try
                    {
                        item.subSequence(i, j);
                        mismatchDescription.appendText(String.format(Locale.ENGLISH, "subSequence(%d, %d) did not throw", i, j));
                        return false;
                    }
                    catch (ArrayIndexOutOfBoundsException | StringIndexOutOfBoundsException e)
                    {
                        // pass
                    }
                }
                else
                {
                    // test valid indexes
                    String expected = mExpectedValue.subSequence(i, j).toString();
                    Matcher subSequenceMatcher = new AllOf<>(
                        new Seq>(
                            hasToString(expected),
                            hasLength(expected.length()),
                            hasChars(expected),
                            hasSubSequences(expected, mSubSequenceTestDepth - 1)));
                    if (!subSequenceMatcher.matches(item.subSequence(i, j)))
                    {
                        mismatchDescription.appendText(String.format(Locale.ENGLISH, "subSequence(%d, %d) ", i, j));
                        subSequenceMatcher.describeMismatch(item.subSequence(i, j), mismatchDescription);
                        return false;
                    }
                }
            }
        }
        return true;
    }


    @Override
    public void describeTo(Description description)
    {
        description.appendText("sub-sequences match \"").appendText(mExpectedValue.toString()).appendText("\"");
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy