GlobMatcher.java

/**************************************************************************
 *
 * Copyright (c) 2016-2020 Yawg project contributors.
 *
 **************************************************************************/

package com.varmateo.yawg.util;

import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.nio.file.Path;
import java.nio.file.PathMatcher;
import java.util.function.Predicate;
import java.util.regex.PatternSyntaxException;

import io.vavr.collection.Array;
import io.vavr.collection.Seq;


/**
 * A predicate for checking if a path matches a given collection of
 * glob patterns.
 */
public final class GlobMatcher
        implements Predicate<Path> {


    private static final FileSystem DEFAULT_FILESYSTEM =
            FileSystems.getDefault();


    private final Seq<String> _globPatterns;
    private final Seq<PathMatcher> _matchers;


    private GlobMatcher(final Builder builder) {

        _globPatterns = builder._globPatterns;
        _matchers = builder._matchers;
    }


    /**
     *
     */
    public static GlobMatcher create(final String... globPatterns) {

        final Builder builder = builder();

        for ( final String globPattern : globPatterns ) {
            builder.addGlobPattern(globPattern);
        }

        return builder.build();
    }


    /**
     * Creates a new builder with no initializations.
     *
     * @return A newly created <code>Builder</code> instance.
     */
    public static Builder builder() {

        return new Builder();
    }


    /**
     * Creates a new builder initialized with the data from the given
     * <code>GlobMatcher</code>.
     *
     * @param data Used for initializing the builder state.
     *
     * @return A newly created <code>Builder</code> instance.
     */
    public static Builder builder(final GlobMatcher data) {

        return new Builder(data);
    }


    /**
     * Checks if the given path matches the glob pattern represented
     * by this object.
     *
     * @param path The path to be checked.
     *
     * @return True if the given path matches our glob pattern. False
     * otherwise.
     */
    @Override
    public boolean test(final Path path) {

        final Path name = path.getFileName();
        final Predicate<PathMatcher> isMatch = matcher -> matcher.matches(name);

        return _matchers
                .filter(isMatch)
                .headOption()
                .isDefined();
    }


    /**
     * Intended for logging or debugging.
     *
     * @return The string representation of our glob pattern.
     */
    @Override
    public String toString() {

        // This implementation is not particularly performant. Let us
        // hope client code only uses this method for sporadic logging
        // or debugging.

        return String.join(",", _globPatterns);
    }





    /**
     * A builder of <code>GlobMatcher</code> instances.
     */
    public static final class Builder {


        private Seq<String> _globPatterns;
        private Seq<PathMatcher> _matchers;


        /**
         *
         */
        /* default */ Builder() {

            _globPatterns = Array.of();
            _matchers = Array.of();
        }


        /**
         *
         */
        /* default */ Builder(final GlobMatcher globMatcher) {

            _globPatterns = globMatcher._globPatterns;
            _matchers = globMatcher._matchers;
        }


        /**
         *
         *
         * @throws PatternSyntaxException If the given glob pattern is
         * invalid.
         */
        public Builder addGlobPattern(final String globPattern) {

            final PathMatcher matcher = DEFAULT_FILESYSTEM.getPathMatcher("glob:" + globPattern);

            _globPatterns = _globPatterns.append(globPattern);
            _matchers = _matchers.append(matcher);

            return this;
        }


        /**
         *
         */
        public Builder addGlobMatcher(final GlobMatcher globMatcher) {

            _globPatterns = _globPatterns.appendAll(globMatcher._globPatterns);
            _matchers = _matchers.appendAll(globMatcher._matchers);

            return this;
        }


        /**
         *
         */
        public GlobMatcher build() {

            return new GlobMatcher(this);
        }


    }


}