Issue #1200 Update deployment manager to use PathWatcher
Updated PathWatcher to use IncludeExcludeSet
This commit is contained in:
parent
578afa0f5e
commit
a947122235
|
@ -39,6 +39,7 @@ import java.nio.file.WatchEvent.Kind;
|
||||||
import java.nio.file.WatchKey;
|
import java.nio.file.WatchKey;
|
||||||
import java.nio.file.WatchService;
|
import java.nio.file.WatchService;
|
||||||
import java.nio.file.attribute.BasicFileAttributes;
|
import java.nio.file.attribute.BasicFileAttributes;
|
||||||
|
import java.util.AbstractSet;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.EventListener;
|
import java.util.EventListener;
|
||||||
|
@ -50,8 +51,10 @@ import java.util.List;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Scanner;
|
import java.util.Scanner;
|
||||||
|
import java.util.Set;
|
||||||
import java.util.concurrent.CopyOnWriteArrayList;
|
import java.util.concurrent.CopyOnWriteArrayList;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
import org.eclipse.jetty.util.component.AbstractLifeCycle;
|
import org.eclipse.jetty.util.component.AbstractLifeCycle;
|
||||||
import org.eclipse.jetty.util.log.Log;
|
import org.eclipse.jetty.util.log.Log;
|
||||||
|
@ -85,16 +88,19 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
}
|
}
|
||||||
|
|
||||||
protected final Path dir;
|
protected final Path dir;
|
||||||
|
protected final IncludeExcludeSet<PathMatcher,Path> includeExclude;
|
||||||
protected int recurseDepth = 0; // 0 means no sub-directories are scanned
|
protected int recurseDepth = 0; // 0 means no sub-directories are scanned
|
||||||
protected List<PathMatcher> includes;
|
|
||||||
protected List<PathMatcher> excludes;
|
|
||||||
protected boolean excludeHidden = false;
|
protected boolean excludeHidden = false;
|
||||||
|
|
||||||
public Config(Path path)
|
public Config(Path path)
|
||||||
|
{
|
||||||
|
this(path,new IncludeExcludeSet<>(PathMatcherSet.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
public Config(Path path, IncludeExcludeSet<PathMatcher,Path> includeExclude)
|
||||||
{
|
{
|
||||||
this.dir = path;
|
this.dir = path;
|
||||||
includes = new ArrayList<>();
|
this.includeExclude = includeExclude;
|
||||||
excludes = new ArrayList<>();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -105,7 +111,7 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
*/
|
*/
|
||||||
public void addExclude(PathMatcher matcher)
|
public void addExclude(PathMatcher matcher)
|
||||||
{
|
{
|
||||||
this.excludes.add(matcher);
|
includeExclude.exclude(matcher);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -189,7 +195,7 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
*/
|
*/
|
||||||
public void addInclude(PathMatcher matcher)
|
public void addInclude(PathMatcher matcher)
|
||||||
{
|
{
|
||||||
this.includes.add(matcher);
|
includeExclude.include(matcher);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -256,9 +262,7 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
*/
|
*/
|
||||||
public Config asSubConfig(Path dir)
|
public Config asSubConfig(Path dir)
|
||||||
{
|
{
|
||||||
Config subconfig = new Config(dir);
|
Config subconfig = new Config(dir,includeExclude);
|
||||||
subconfig.includes = this.includes;
|
|
||||||
subconfig.excludes = this.excludes;
|
|
||||||
if (dir == this.dir)
|
if (dir == this.dir)
|
||||||
subconfig.recurseDepth = this.recurseDepth; // TODO shouldn't really do a subconfig for this
|
subconfig.recurseDepth = this.recurseDepth; // TODO shouldn't really do a subconfig for this
|
||||||
else
|
else
|
||||||
|
@ -286,18 +290,7 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
return this.dir;
|
return this.dir;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean hasMatch(Path path, List<PathMatcher> matchers)
|
@Deprecated
|
||||||
{
|
|
||||||
for (PathMatcher matcher : matchers)
|
|
||||||
{
|
|
||||||
if (matcher.matches(path))
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isExcluded(Path dir) throws IOException
|
public boolean isExcluded(Path dir) throws IOException
|
||||||
{
|
{
|
||||||
if (excludeHidden)
|
if (excludeHidden)
|
||||||
|
@ -312,13 +305,7 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (excludes.isEmpty())
|
boolean matched = ((PathMatcherSet)includeExclude.getExcluded()).test(dir);
|
||||||
{
|
|
||||||
// no excludes == everything allowed
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean matched = hasMatch(dir,excludes);
|
|
||||||
if (NOISY_LOG.isDebugEnabled())
|
if (NOISY_LOG.isDebugEnabled())
|
||||||
{
|
{
|
||||||
NOISY_LOG.debug("isExcluded [{}] on {}",matched,dir);
|
NOISY_LOG.debug("isExcluded [{}] on {}",matched,dir);
|
||||||
|
@ -326,19 +313,10 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
return matched;
|
return matched;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Deprecated
|
||||||
public boolean isIncluded(Path dir)
|
public boolean isIncluded(Path dir)
|
||||||
{
|
{
|
||||||
if (includes.isEmpty())
|
boolean matched = ((PathMatcherSet)includeExclude.getIncluded()).test(dir);
|
||||||
{
|
|
||||||
// no includes == everything allowed
|
|
||||||
if (NOISY_LOG.isDebugEnabled())
|
|
||||||
{
|
|
||||||
NOISY_LOG.debug("isIncluded [All] on {}",dir);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean matched = hasMatch(dir,includes);
|
|
||||||
if (NOISY_LOG.isDebugEnabled())
|
if (NOISY_LOG.isDebugEnabled())
|
||||||
{
|
{
|
||||||
NOISY_LOG.debug("isIncluded [{}] on {}",matched,dir);
|
NOISY_LOG.debug("isIncluded [{}] on {}",matched,dir);
|
||||||
|
@ -348,15 +326,7 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
|
|
||||||
public boolean matches(Path path)
|
public boolean matches(Path path)
|
||||||
{
|
{
|
||||||
try
|
return includeExclude.test(path);
|
||||||
{
|
|
||||||
return !isExcluded(path) && isIncluded(path);
|
|
||||||
}
|
|
||||||
catch (IOException e)
|
|
||||||
{
|
|
||||||
LOG.warn("Unable to match path: " + path,e);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1453,4 +1423,17 @@ public class PathWatcher extends AbstractLifeCycle implements Runnable
|
||||||
appendConfigId(s);
|
appendConfigId(s);
|
||||||
return s.toString();
|
return s.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static class PathMatcherSet extends HashSet<PathMatcher> implements Predicate<Path>
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public boolean test(Path path)
|
||||||
|
{
|
||||||
|
for (PathMatcher pm: this)
|
||||||
|
if (pm.matches(path))
|
||||||
|
return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -273,13 +273,16 @@ public class PathWatcherTest
|
||||||
*/
|
*/
|
||||||
private static void awaitQuietTime(PathWatcher pathWatcher) throws InterruptedException
|
private static void awaitQuietTime(PathWatcher pathWatcher) throws InterruptedException
|
||||||
{
|
{
|
||||||
double multiplier = 5.0;
|
long offset = 1000;
|
||||||
|
double multiplier = 3.0;
|
||||||
if (OS.IS_WINDOWS)
|
if (OS.IS_WINDOWS)
|
||||||
{
|
{
|
||||||
// Microsoft Windows filesystem is too slow for a lower multiplier
|
// Microsoft Windows filesystem is too slow for a lower multiplier
|
||||||
multiplier = 6.0;
|
offset = 1500;
|
||||||
|
multiplier = 4.0;
|
||||||
}
|
}
|
||||||
TimeUnit.MILLISECONDS.sleep((long)((double)pathWatcher.getUpdateQuietTimeMillis() * multiplier));
|
long wait = offset + (long)((double)pathWatcher.getUpdateQuietTimeMillis() * multiplier);
|
||||||
|
TimeUnit.MILLISECONDS.sleep(wait);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static final int KB = 1024;
|
private static final int KB = 1024;
|
||||||
|
@ -393,6 +396,7 @@ public class PathWatcherTest
|
||||||
expected.put("a.txt",new PathWatchEventType[] {ADDED});
|
expected.put("a.txt",new PathWatchEventType[] {ADDED});
|
||||||
expected.put("b.txt",new PathWatchEventType[] {ADDED});
|
expected.put("b.txt",new PathWatchEventType[] {ADDED});
|
||||||
|
|
||||||
|
Thread.currentThread().sleep(1000); // TODO poor test
|
||||||
|
|
||||||
capture.assertEvents(expected);
|
capture.assertEvents(expected);
|
||||||
|
|
||||||
|
@ -401,7 +405,7 @@ public class PathWatcherTest
|
||||||
|
|
||||||
capture.reset();
|
capture.reset();
|
||||||
|
|
||||||
Thread.currentThread().sleep(1000);
|
Thread.currentThread().sleep(1000); // TODO poor test
|
||||||
|
|
||||||
pathWatcher.start();
|
pathWatcher.start();
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue