View Javadoc
1   /*
2    * Licensed to the Apache Software Foundation (ASF) under one or more
3    * contributor license agreements.  See the NOTICE file distributed with
4    * this work for additional information regarding copyright ownership.
5    * The ASF licenses this file to You under the Apache License, Version 2.0
6    * (the "License"); you may not use this file except in compliance with
7    * the License.  You may obtain a copy of the License at
8    *
9    *      http://www.apache.org/licenses/LICENSE-2.0
10   *
11   * Unless required by applicable law or agreed to in writing, software
12   * distributed under the License is distributed on an "AS IS" BASIS,
13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14   * See the License for the specific language governing permissions and
15   * limitations under the License.
16   */
17  package org.apache.commons.io.filefilter;
18  
19  import java.io.File;
20  import java.io.Serializable;
21  import java.nio.file.FileVisitResult;
22  import java.nio.file.Path;
23  import java.nio.file.attribute.BasicFileAttributes;
24  import java.util.ArrayList;
25  import java.util.Collections;
26  import java.util.List;
27  import java.util.Objects;
28  import java.util.stream.Stream;
29  
30  /**
31   * A {@link java.io.FileFilter} providing conditional AND logic across a list of
32   * file filters. This filter returns {@code true} if all filters in the
33   * list return {@code true}. Otherwise, it returns {@code false}.
34   * Checking of the file filter list stops when the first filter returns
35   * {@code false}.
36   * <h2>Deprecating Serialization</h2>
37   * <p>
38   * <em>Serialization is deprecated and will be removed in 3.0.</em>
39   * </p>
40   *
41   * @since 1.0
42   * @see FileFilterUtils#and(IOFileFilter...)
43   */
44  public class AndFileFilter extends AbstractFileFilter implements ConditionalFileFilter, Serializable {
45  
46      private static final long serialVersionUID = 7215974688563965257L;
47  
48      /** The list of file filters. */
49      private final List<IOFileFilter> fileFilters;
50  
51      /**
52       * Constructs a new empty instance.
53       *
54       * @since 1.1
55       */
56      public AndFileFilter() {
57          this(0);
58      }
59  
60      /**
61       * Constructs a new instance with the given initial list.
62       *
63       * @param initialList the initial list.
64       */
65      private AndFileFilter(final ArrayList<IOFileFilter> initialList) {
66          this.fileFilters = Objects.requireNonNull(initialList, "initialList");
67      }
68  
69      /**
70       * Constructs a new instance with the given initial capacity.
71       *
72       * @param initialCapacity the initial capacity.
73       */
74      private AndFileFilter(final int initialCapacity) {
75          this(new ArrayList<>(initialCapacity));
76      }
77  
78      /**
79       * Constructs a new instance for the give filters.
80       *
81       * @param fileFilters filters to OR.
82       * @since 2.9.0
83       */
84      public AndFileFilter(final IOFileFilter... fileFilters) {
85          this(Objects.requireNonNull(fileFilters, "fileFilters").length);
86          addFileFilter(fileFilters);
87      }
88  
89      /**
90       * Constructs a new file filter that ANDs the result of other filters.
91       *
92       * @param filter1  the first filter, must second be null
93       * @param filter2  the first filter, must not be null
94       * @throws IllegalArgumentException if either filter is null
95       */
96      public AndFileFilter(final IOFileFilter filter1, final IOFileFilter filter2) {
97          this(2);
98          addFileFilter(filter1);
99          addFileFilter(filter2);
100     }
101 
102     /**
103      * Constructs a new instance of {@link AndFileFilter}
104      * with the specified list of filters.
105      *
106      * @param fileFilters  a List of IOFileFilter instances, copied.
107      * @since 1.1
108      */
109     public AndFileFilter(final List<IOFileFilter> fileFilters) {
110         this(new ArrayList<>(Objects.requireNonNull(fileFilters, "fileFilters")));
111     }
112 
113     /**
114      * {@inheritDoc}
115      */
116     @Override
117     public boolean accept(final File file) {
118         return !isEmpty() && fileFilters.stream().allMatch(fileFilter -> fileFilter.accept(file));
119     }
120 
121     /**
122      * {@inheritDoc}
123      */
124     @Override
125     public boolean accept(final File file, final String name) {
126         return !isEmpty() && fileFilters.stream().allMatch(fileFilter -> fileFilter.accept(file, name));
127     }
128 
129     /**
130      * {@inheritDoc}
131      * @since 2.9.0
132      */
133     @Override
134     public FileVisitResult accept(final Path file, final BasicFileAttributes attributes) {
135         return isEmpty() ? FileVisitResult.TERMINATE
136                 : toDefaultFileVisitResult(fileFilters.stream().allMatch(fileFilter -> fileFilter.accept(file, attributes) == FileVisitResult.CONTINUE));
137     }
138 
139     /**
140      * {@inheritDoc}
141      */
142     @Override
143     public void addFileFilter(final IOFileFilter fileFilter) {
144         fileFilters.add(Objects.requireNonNull(fileFilter, "fileFilter"));
145     }
146 
147     /**
148      * Adds the given file filters.
149      *
150      * @param fileFilters the filters to add.
151      * @since 2.9.0
152      */
153     public void addFileFilter(final IOFileFilter... fileFilters) {
154         Stream.of(Objects.requireNonNull(fileFilters, "fileFilters")).forEach(this::addFileFilter);
155     }
156 
157     /**
158      * {@inheritDoc}
159      */
160     @Override
161     public List<IOFileFilter> getFileFilters() {
162         return Collections.unmodifiableList(fileFilters);
163     }
164 
165     private boolean isEmpty() {
166         return fileFilters.isEmpty();
167     }
168 
169     /**
170      * {@inheritDoc}
171      */
172     @Override
173     public boolean removeFileFilter(final IOFileFilter ioFileFilter) {
174         return fileFilters.remove(ioFileFilter);
175     }
176 
177     /**
178      * {@inheritDoc}
179      */
180     @Override
181     public void setFileFilters(final List<IOFileFilter> fileFilters) {
182         this.fileFilters.clear();
183         this.fileFilters.addAll(fileFilters);
184     }
185 
186     /**
187      * Builds a String representation of this file filter.
188      *
189      * @return a String representation
190      */
191     @Override
192     public String toString() {
193         final StringBuilder buffer = new StringBuilder();
194         buffer.append(super.toString());
195         buffer.append("(");
196         append(fileFilters, buffer);
197         buffer.append(")");
198         return buffer.toString();
199     }
200 
201 }