com.metreeca.mark.tasks.Watch Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of mark-maven-plugin Show documentation
Show all versions of mark-maven-plugin Show documentation
A minimalist Maven plugin for static site generation
/*
* Copyright © 2019-2020 Metreeca srl
*
* 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 com.metreeca.mark.tasks;
import com.metreeca.mark.*;
import com.sun.nio.file.SensitivityWatchEventModifier;
import org.apache.maven.plugin.logging.Log;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.nio.file.*;
import java.util.function.Consumer;
import java.util.stream.Stream;
import static java.nio.file.StandardWatchEventKinds.*;
/**
* Site watching task.
*
* Generates a processed version of the {@linkplain Opts#source() source} site folder in the
* {@linkplain Opts#target() target} site folder, watching the {@linkplain Opts#source() source} site folder for
* further changes to by synchronized.
*/
public final class Watch implements Task {
@Override public void exec(final Mark mark) {
final Path source=mark.source();
final Path assets=mark.assets();
final Log logger=mark.logger();
try ( final WatchService service=source.getFileSystem().newWatchService() ) {
final Consumer register=path -> {
try {
path.register(service,
new WatchEvent.Kind>[]{ ENTRY_CREATE, ENTRY_MODIFY },
SensitivityWatchEventModifier.HIGH
);
} catch ( final IOException e ) {
throw new UncheckedIOException(e);
}
};
try ( final Stream sources=Files.walk(source) ) {
sources.filter(Files::isDirectory).forEach(register); // register existing source folders
}
if ( source.getFileSystem().equals(assets.getFileSystem()) ) { // ignore bundled assets
try ( final Stream assetses=Files.walk(assets) ) {
assetses.filter(Files::isDirectory).forEach(register); // register existing assets folders
}
}
for (WatchKey key; (key=service.take()) != null; key.reset()) { // watch source changes
for (final WatchEvent> event : key.pollEvents()) {
final WatchEvent.Kind> kind=event.kind();
final Path path=((Path)key.watchable()).resolve((Path)event.context());
if ( event.kind().equals(ENTRY_CREATE) && Files.isDirectory(path) ) { // register new folders
logger.info(source.relativize(path).toString());
register.accept(path);
} else if ( event.kind().equals(ENTRY_CREATE) && Files.isRegularFile(path) ) {
mark.process(Stream.of(path));
} else if ( event.kind().equals(ENTRY_MODIFY) && Files.isRegularFile(path) ) {
if ( mark.isLayout(path) ) { mark.exec(new Build()); } else { mark.process(Stream.of(path)); }
} else if ( kind.equals(OVERFLOW) ) {
logger.error("sync lost ;-(");
}
}
}
} catch ( final IOException e ) {
throw new UncheckedIOException(e);
} catch ( final InterruptedException ignored ) {
logger.error("interrupted…");
}
}
}