DirBaker.java

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

package com.varmateo.yawg.core;

import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.function.Function;

import io.vavr.collection.Seq;

import com.varmateo.yawg.api.YawgException;
import com.varmateo.yawg.logging.Log;
import com.varmateo.yawg.logging.LogFactory;
import com.varmateo.yawg.spi.DirBakeListener;
import com.varmateo.yawg.spi.PageContext;
import com.varmateo.yawg.spi.TemplateService;


/**
 * Bakes the files contained in a directory, and recursively bake all
 * sub-directories.
 */
/* default */ final class DirBaker {


    private final Log _log;
    private final FileBaker _fileBaker;
    private final DirBakeOptionsDao _dirBakeOptionsDao;
    private final DirBakeListener _listener;
    private final TemplateService _templateService;


    /**
     * @param log Used for logging.
     *
     * @param fileBaker Baker to be used on regular files.
     *
     * @param templateService Will provide the templates used in the
     * baking of individual files.
     *
     * @param dirBakeOptionsDao Used for reading the bake configuration
     * for each directory.
     *
     * @param dirBakeListener Will be notified when the bake of a
     * directory starts.
     */
    /* default */ DirBaker(
            final FileBaker fileBaker,
            final TemplateService templateService,
            final DirBakeOptionsDao dirBakeOptionsDao,
            final DirBakeListener dirBakeListener) {

        _log = LogFactory.createFor(DirBaker.class);
        _fileBaker = fileBaker;
        _dirBakeOptionsDao = dirBakeOptionsDao;
        _listener = dirBakeListener;
        _templateService = templateService;
    }


    /**
     * 
     */
    public void bakeDirectory(
            final Path sourceDir,
            final Path targetDir,
            final DirBakeOptions parentDirBakeOptions) {

        final DirBakerContext context = DirBakerContext.create(
                sourceDir, targetDir, _templateService, parentDirBakeOptions);

        doBakeDirectory(sourceDir, targetDir, context);
    }


    /**
     * 
     */
    private void doBakeDirectory(
            final Path sourceDir,
            final Path targetDir,
            final DirBakerContext parentContext) {

        final Path relSourceDir = parentContext.sourceRootDir().relativize(sourceDir);
        _log.debug("Baking directory {0}", relSourceDir);

        createDirIfNeeded(targetDir);

        final DirBakeOptions specificDirBakeOptions = _dirBakeOptionsDao.loadFromDir(sourceDir);
        final DirBakerContext context = parentContext.buildForChildDir(
                targetDir,
                specificDirBakeOptions,
                x -> _listener.onDirBake(x).pageVars());
        final DirBakeOptions dirBakeOptions = context.dirBakeOptions();
        final Seq<Path> dirEntries = getDirEntries(sourceDir, dirBakeOptions);

        bakeChildFiles(dirEntries, targetDir, dirBakeOptions, context.pageContext());
        bakeChildDirectories(dirEntries, targetDir, context);
        bakeExtraDirectories(sourceDir, targetDir, context);
    }


    /**
     *
     */
    private void createDirIfNeeded(final Path targetDir) {

        if ( !Files.exists(targetDir) ) {
            doIoAction(
                    () -> Files.createDirectory(targetDir),
                    DirBakerException.directoryCreationFailure(targetDir));
        }
    }


    /**
     *
     */
    private Seq<Path> getDirEntries(
            final Path dir,
            final DirBakeOptions dirBakeOptions) {

        final DirEntryScanner scanner = new DirEntryScanner(dirBakeOptions);

        return doIoAction(
                () -> scanner.getDirEntries(dir),
                DirBakerException.directoryListFailure(dir));
    }


    /**
     *
     */
    private void bakeChildFiles(
            final Seq<Path> dirEntries,
            final Path targetDir,
            final DirBakeOptions dirBakeOptions,
            final PageContext context) {

        final Seq<Path> filePathList = dirEntries.filter(Files::isRegularFile);

        for ( final Path path : filePathList ) {
            _fileBaker.bakeFile(path, context, targetDir, dirBakeOptions);
        }
    }


    /**
     *
     */
    private void bakeChildDirectories(
            final Seq<Path> dirEntries,
            final Path targetDir,
            final DirBakerContext context) {

        final Seq<Path> dirPathList = dirEntries.filter(Files::isDirectory);

        for ( final Path childSourceDir : dirPathList ) {
            final Path dirBasename = childSourceDir.getFileName();
            final Path childTargetDir = targetDir.resolve(dirBasename);

            doBakeDirectory(childSourceDir, childTargetDir, context);
        }
    }


    /**
     *
     */
    private void bakeExtraDirectories(
            final Path sourceDir,
            final Path targetDir,
            final DirBakerContext context) {

        final Seq<Path> extraDirPathList = context.dirBakeOptions()
                .extraDirsHere
                .map(path -> sourceDir.resolve(path));

        for ( final Path extraSourceDir : extraDirPathList ) {
            doBakeDirectory(extraSourceDir, targetDir, context);
        }
    }


    /**
     * A simple wrapper to change a method signature from the checked
     * IOException to the unchecked YawgException.
     */
    private <T> T doIoAction(
            final IoSupplier<T> supplier,
            final Function<IOException, DirBakerException> exceptionBuilder) {

        final T result;

        try {
            result = supplier.get();
        } catch ( IOException e ) {
            throw exceptionBuilder.apply(e);
        }

        return result;
    }


    /**
     * Utility interface to simplify creating lambdas whose body may
     * throw exceptions.
     *
     * @param <T> Type of the values returned by the function.
     */
    @FunctionalInterface
    private interface IoSupplier<T> {


        /**
         *
         */
        T get() throws IOException;
    }


}