com.yahoo.config.model.application.provider.FilesApplicationPackage Maven / Gradle / Ivy
The newest version!
// Copyright Vespa.ai. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package com.yahoo.config.model.application.provider;
import com.yahoo.component.Version;
import com.yahoo.component.Vtag;
import com.yahoo.config.application.ConfigDefinitionDir;
import com.yahoo.config.application.Xml;
import com.yahoo.config.application.XmlPreProcessor;
import com.yahoo.config.application.api.ApplicationFile;
import com.yahoo.config.application.api.ApplicationMetaData;
import com.yahoo.config.application.api.ApplicationPackage;
import com.yahoo.config.application.api.ComponentInfo;
import com.yahoo.config.application.api.DeployLogger;
import com.yahoo.config.application.api.DeploymentSpec;
import com.yahoo.config.application.api.UnparsedConfigDefinition;
import com.yahoo.config.codegen.DefParser;
import com.yahoo.config.model.application.AbstractApplicationPackage;
import com.yahoo.config.provision.ApplicationId;
import com.yahoo.config.provision.ApplicationName;
import com.yahoo.config.provision.InstanceName;
import com.yahoo.config.provision.TenantName;
import com.yahoo.config.provision.Zone;
import com.yahoo.io.HexDump;
import com.yahoo.io.IOUtils;
import com.yahoo.io.reader.NamedReader;
import com.yahoo.path.Path;
import com.yahoo.text.Utf8;
import com.yahoo.text.XML;
import com.yahoo.vespa.config.ConfigDefinition;
import com.yahoo.vespa.config.ConfigDefinitionBuilder;
import com.yahoo.vespa.config.ConfigDefinitionKey;
import com.yahoo.vespa.config.util.ConfigUtils;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.xml.sax.SAXException;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.Reader;
import java.io.StringReader;
import java.nio.file.AccessDeniedException;
import java.nio.file.DirectoryNotEmptyException;
import java.nio.file.Files;
import java.security.MessageDigest;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.function.BiFunction;
import java.util.logging.Level;
import java.util.logging.Logger;
import static com.yahoo.text.Lowercase.toLowerCase;
import static com.yahoo.yolean.Exceptions.uncheck;
/**
* Application package derived from local files, i.e. during deploy.
* Construct using {@link com.yahoo.config.model.application.provider.FilesApplicationPackage#fromFile(java.io.File)} or
* {@link com.yahoo.config.model.application.provider.FilesApplicationPackage#fromFileWithDeployData(java.io.File, DeployData)}.
*
* @author Vegard Havdal
*/
public class FilesApplicationPackage extends AbstractApplicationPackage {
/**
* The name of the subdirectory (below the original application package root)
* where a preprocessed version of this application package is stored.
* As it happens, the config model is first created with an application package in this subdirectory,
* and later backed by an application package which is not in this subdirectory.
* To enable model code to correct for this, this constant must be publicly known.
*
* All of this stuff is Very Unfortunate and should be fixed. -Jon
*/
public static final String preprocessed = ".preprocessed";
private static final Logger log = Logger.getLogger(FilesApplicationPackage.class.getName());
private static final String META_FILE_NAME = ".applicationMetaData";
private static final Map> validFileExtensions;
private final File appDir;
private final File preprocessedDir;
private final File configDefsDir;
private final AppSubDirs appSubDirs;
// NOTE: these directories exist in the original user app, but their locations are given in 'services.xml'
private final List userIncludeDirs = new ArrayList<>();
private final ApplicationMetaData metaData;
private final boolean includeSourceFiles;
private final TransformerFactory transformerFactory;
private DeploymentSpec deploymentSpec = null;
/** Creates from a directory with source files included */
public static FilesApplicationPackage fromFile(File appDir) {
return fromFile(appDir, false);
}
/**
* Returns an application package object based on the given application dir
*
* @param appDir application package directory
* @param includeSourceFiles read files from source directories /src/main and src/test in addition
* to the application package location. This is useful during development
* to be able to run tests without a complete build first.
* @return an Application package instance
*/
public static FilesApplicationPackage fromFile(File appDir, boolean includeSourceFiles) {
return new Builder(appDir).preprocessedDir(applicationFile(appDir, preprocessed))
.includeSourceFiles(includeSourceFiles)
.build();
}
/** Creates package from a local directory, typically deploy app */
public static FilesApplicationPackage fromFileWithDeployData(File appDir, DeployData deployData) {
return fromFileWithDeployData(appDir, deployData, false);
}
/** Creates package from a local directory, typically deploy app */
public static FilesApplicationPackage fromFileWithDeployData(File appDir,
DeployData deployData,
boolean includeSourceFiles) {
return new Builder(appDir).includeSourceFiles(includeSourceFiles).deployData(deployData).build();
}
private static ApplicationMetaData metaDataFromDeployData(File appDir, DeployData deployData) {
return new ApplicationMetaData(deployData.getDeployTimestamp(),
deployData.isInternalRedeploy(),
deployData.getApplicationId(),
computeCheckSum(appDir),
deployData.getGeneration(),
deployData.getCurrentlyActiveGeneration());
}
/**
* New package from given path on local file system. Retrieves config definition files from
* the default location '$VESPA_HOME/share/vespa/configdefinitions'.
*
* @param appDir application package directory
* @param preprocessedDir preprocessed application package output directory
* @param metaData metadata for this application package
* @param includeSourceFiles include files from source dirs
*/
private FilesApplicationPackage(File appDir, File preprocessedDir, ApplicationMetaData metaData, boolean includeSourceFiles) {
verifyAppDir(appDir);
this.includeSourceFiles = includeSourceFiles;
this.appDir = appDir;
this.preprocessedDir = preprocessedDir;
appSubDirs = new AppSubDirs(appDir);
configDefsDir = applicationFile(appDir, CONFIG_DEFINITIONS_DIR);
addUserIncludeDirs();
this.metaData = metaData;
this.transformerFactory = XML.createTransformerFactory();
}
@Override
public ApplicationId getApplicationId() { return metaData.getApplicationId(); }
@Override
public List getFiles(Path relativePath, String suffix, boolean recurse) {
return getFiles(relativePath, "", suffix, recurse);
}
@Override
public ApplicationFile getFile(Path path) {
File file = (path.isRoot() ? appDir : applicationFile(appDir, path.getRelative()));
return new FilesApplicationFile(path, file);
}
@Override
public ApplicationMetaData getMetaData() {
return metaData;
}
private List getFiles(Path relativePath, String namePrefix, String suffix, boolean recurse) {
try {
List readers=new ArrayList<>();
File dir = applicationFile(appDir, relativePath);
if ( ! dir.isDirectory()) return readers;
File[] files = dir.listFiles();
if (files != null) {
for (File file : files) {
if (file.isDirectory()) {
if (recurse)
readers.addAll(getFiles(relativePath.append(file.getName()), namePrefix + "/" + file.getName(), suffix, recurse));
} else {
if (suffix == null || file.getName().endsWith(suffix))
readers.add(new NamedReader(file.getName(), new FileReader(file)));
}
}
}
return readers;
}
catch (IOException e) {
throw new RuntimeException("Could not open (all) files in '" + relativePath + "'",e);
}
}
private void verifyAppDir(File appDir) {
Objects.requireNonNull(appDir, "Path cannot be null");
if ( ! appDir.exists()) {
throw new IllegalArgumentException("Path '" + appDir + "' does not exist");
}
if ( ! appDir.isDirectory()) {
throw new IllegalArgumentException("Path '" + appDir + "' is not a directory");
}
if (! appDir.canRead()){
throw new IllegalArgumentException("Cannot read from application directory '" + appDir + "'");
}
}
@Override
public Reader getHosts() {
try {
File hostsFile = getHostsFile();
if (!hostsFile.exists()) return null;
return new FileReader(hostsFile);
} catch (Exception e) {
throw new IllegalArgumentException(e);
}
}
@Override
public String getHostSource() {
return getHostsFile().getPath();
}
private File getHostsFile() {
return applicationFile(appDir, HOSTS);
}
@Override
public String getServicesSource() {
return getServicesFile().getPath();
}
private File getServicesFile() {
return applicationFile(appDir, SERVICES);
}
@Override
public Optional getDeployment() { return optionalFile(DEPLOYMENT_FILE); }
@Override
public Optional getValidationOverrides() { return optionalFile(VALIDATION_OVERRIDES); }
private Optional optionalFile(Path filePath) {
try {
return Optional.of(getFile(filePath).createReader());
} catch (FileNotFoundException e) {
return Optional.empty();
}
}
@Override
public List getUserIncludeDirs() {
return Collections.unmodifiableList(userIncludeDirs);
}
public void addUserIncludeDirs() {
Document services;
try {
services = Xml.getDocument(getServices());
} catch (Exception e) {
return; // This method does not validate that services.xml exists, or that it is valid xml.
}
NodeList includeNodes = services.getElementsByTagName(IncludeDirs.INCLUDE);
for (int i=0; i < includeNodes.getLength(); i++) {
Node includeNode = includeNodes.item(i);
addIncludeDir(includeNode);
}
}
private void addIncludeDir(Node includeNode) {
if (! (includeNode instanceof Element))
return;
Element include = (Element) includeNode;
if (! include.hasAttribute(IncludeDirs.DIR))
return;
String dir = include.getAttribute(IncludeDirs.DIR);
validateIncludeDir(dir);
IncludeDirs.validateFilesInIncludedDir(dir, include.getParentNode(), this);
log.log(Level.FINE, () -> "Adding user include dir '" + dir + "'");
userIncludeDirs.add(dir);
}
@Override
public void validateIncludeDir(String dirName) {
IncludeDirs.validateIncludeDir(dirName, this);
}
@Override
public Collection getSchemas() {
Set ret = new LinkedHashSet<>();
try {
for (File f : getSearchDefinitionFiles()) {
ret.add(new NamedReader(f.getName(), new FileReader(f)));
}
} catch (Exception e) {
throw new IllegalArgumentException("Couldn't get schema contents.", e);
}
return ret;
}
/**
* Creates a reader for a config definition
*
* @param defPath the path to the application package
* @return the reader of this config definition
*/
private Reader retrieveConfigDefReader(File defPath) {
try {
return new NamedReader(defPath.getPath(), new FileReader(defPath));
} catch (IOException e) {
throw new IllegalArgumentException("Could not read config definition file '" + defPath + "'", e);
}
}
@Override
public Map getAllExistingConfigDefs() {
Map defs = new LinkedHashMap<>();
addAllDefsFromConfigDir(defs, configDefsDir);
if (includeSourceFiles) { // allow running from source, assuming mvn file project layout
addAllDefsFromConfigDir(defs, new File("src/main/resources/configdefinitions"));
addAllDefsFromConfigDir(defs, new File("src/test/resources/configdefinitions"));
}
addAllDefsFromBundles(defs, getComponents(appDir));
return defs;
}
private void addAllDefsFromBundles(Map defs, List components) {
for (Component component : components) {
Bundle bundle = component.getBundle();
for (final Bundle.DefEntry def : bundle.getDefEntries()) {
final ConfigDefinitionKey defKey = new ConfigDefinitionKey(def.defName, def.defNamespace);
if (!defs.containsKey(defKey)) {
defs.put(defKey, new UnparsedConfigDefinition() {
@Override
public ConfigDefinition parse() {
DefParser parser = new DefParser(defKey.getName(), new StringReader(def.contents));
return ConfigDefinitionBuilder.createConfigDefinition(parser.getTree());
}
@Override
public String getUnparsedContent() {
return def.contents;
}
});
}
}
}
}
private void addAllDefsFromConfigDir(Map defs, File configDefsDir) {
if (! configDefsDir.isDirectory()) return;
log.log(Level.FINE, () -> "Getting all config definitions from '" + configDefsDir + "'");
for (File def : configDefsDir.listFiles((File dir, String name) -> name.matches(".*\\.def"))) {
String[] nv = def.getName().split("\\.def");
ConfigDefinitionKey key;
try {
key = ConfigUtils.createConfigDefinitionKeyFromDefFile(def);
} catch (IOException e) {
e.printStackTrace(); // TODO: Fix
break;
}
if (key.getNamespace().isEmpty())
throw new IllegalArgumentException("Config definition '" + def + "' has no namespace");
if (defs.containsKey(key)) {
if (nv[0].contains(".")) {
log.log(Level.INFO, "Two config definitions found for the same name and namespace: " + key +
". The file '" + def + "' will take precedence");
} else {
log.log(Level.INFO, "Two config definitions found for the same name and namespace: " + key +
". Skipping '" + def + "', as it does not contain namespace in filename");
continue; // skip
}
}
defs.put(key, new UnparsedConfigDefinition() {
@Override
public ConfigDefinition parse() {
DefParser parser = new DefParser(key.getName(), retrieveConfigDefReader(def));
return ConfigDefinitionBuilder.createConfigDefinition(parser.getTree());
}
@Override
public String getUnparsedContent() {
return readConfigDefinition(def);
}
});
}
}
private String readConfigDefinition(File defPath) {
try (Reader reader = retrieveConfigDefReader(defPath)) {
return IOUtils.readAll(reader);
} catch (IOException e) {
throw new RuntimeException("Error reading config definition '" + defPath + "'", e);
}
}
@Override
public Reader getServices() {
try {
return new FileReader(getServicesSource());
} catch (Exception e) {
throw new IllegalArgumentException(e);
}
}
static List getSearchDefinitionFiles(File appDir) {
List schemaFiles = new ArrayList<>();
File sdDir = applicationFile(appDir, SEARCH_DEFINITIONS_DIR.getRelative());
if (sdDir.isDirectory())
schemaFiles.addAll(List.of(sdDir.listFiles((dir, name) -> validSchemaFilename(name))));
sdDir = applicationFile(appDir, SCHEMAS_DIR.getRelative());
if (sdDir.isDirectory())
schemaFiles.addAll(List.of(sdDir.listFiles((dir, name) -> validSchemaFilename(name))));
return schemaFiles;
}
public List getSearchDefinitionFiles() {
return getSearchDefinitionFiles(appDir);
}
// Only for use by deploy processor
public static List getComponents(File appDir) {
return components(appDir, Component::new);
}
private static List getComponentsInfo(File appDir) {
return components(appDir, (__, info) -> info);
}
private static List components(File appDir, BiFunction toValue) {
List components = new ArrayList<>();
for (Bundle bundle : Bundle.getBundles(applicationFile(appDir, COMPONENT_DIR))) {
components.add(toValue.apply(bundle, new ComponentInfo(Path.fromString(COMPONENT_DIR).append(bundle.getFile().getName()).getRelative())));
}
return components;
}
@Override
public List getComponentsInfo(Version vespaVersion) {
return getComponentsInfo(appDir);
}
/**
* Returns a list of all components in this package.
*
* @return A list of components.
*/
public List getComponents() {
return getComponents(appDir);
}
public File getAppDir() throws IOException {
return appDir.getCanonicalFile();
}
private static ApplicationMetaData readMetaData(File appDir) {
String originalAppDir = preprocessed.equals(appDir.getName()) ? appDir.getParentFile().getName() : appDir.getName();
ApplicationMetaData defaultMetaData = new ApplicationMetaData(0L,
false,
ApplicationId.from(TenantName.defaultName(),
ApplicationName.from(originalAppDir),
InstanceName.defaultName()),
"",
0L,
0L);
File metaFile = applicationFile(appDir, META_FILE_NAME);
if ( ! metaFile.exists()) {
return defaultMetaData;
}
try (FileReader reader = new FileReader(metaFile)) {
return ApplicationMetaData.fromJsonString(IOUtils.readAll(reader));
} catch (Exception e) {
// Not a big deal, return default
return defaultMetaData;
}
}
/**
* Represents a component in the application package. Immutable.
*/
public static class Component {
public final ComponentInfo info;
private final Bundle bundle;
public Component(Bundle bundle, ComponentInfo info) {
this.bundle = bundle;
this.info = info;
}
public List getDefEntries() {
return bundle.getDefEntries();
}
public Bundle getBundle() {
return bundle;
}
}
/**
* Reads a ranking expression from file to a string and returns it.
*
* @param name the name of the file to return,
* relative to the search definition directory in the application package
* @return the content of a ranking expression file
* @throws IllegalArgumentException if the file was not found or could not be read
*/
@Override
public Reader getRankingExpression(String name) {
try {
return IOUtils.createReader(expressionFileNameToFile(name), "utf-8");
}
catch (IOException e) {
throw new IllegalArgumentException("Could not read ranking expression file '" + name + "'", e);
}
}
private File expressionFileNameToFile(String name) {
if (new File(name).isAbsolute())
throw new IllegalArgumentException("Absolute path to ranking expression file is not allowed: " + name);
Path path = Path.fromString(name);
File expressionFile = applicationFile(appDir, SCHEMAS_DIR.append(path));
if ( ! expressionFile.exists()) {
expressionFile = applicationFile(appDir, SEARCH_DEFINITIONS_DIR.append(path));
}
return expressionFile;
}
@Override
public File getFileReference(Path pathRelativeToAppDir) {
return applicationFile(appDir, pathRelativeToAppDir.getRelative());
}
@Override
public void validateXML() throws IOException {
validateXMLFor(Optional.empty());
}
@Override
public void validateXMLFor(Optional vespaVersion) throws IOException {
Version modelVersion = vespaVersion.orElse(Vtag.currentVersion);
ApplicationPackageXmlFilesValidator validator = ApplicationPackageXmlFilesValidator.create(appDir, modelVersion);
validator.checkApplication();
validator.checkIncludedDirs(this);
}
@Override
public void writeMetaData() {
File metaFile = applicationFile(appDir, META_FILE_NAME);
IOUtils.writeFile(metaFile, metaData.asJsonBytes());
}
@Override
public DeploymentSpec getDeploymentSpec() {
if (deploymentSpec != null) return deploymentSpec;
return deploymentSpec = parseDeploymentSpec(false);
}
private void preprocessXML(File destination, File inputXml, Zone zone) throws IOException {
if ( ! inputXml.exists()) return;
try {
InstanceName instance = metaData.getApplicationId().instance();
Document document = new XmlPreProcessor(appDir,
inputXml,
instance,
zone.environment(),
zone.region(),
zone.cloud().name(),
getDeploymentSpec().tags(instance, zone.environment()))
.run();
try (FileOutputStream outputStream = new FileOutputStream(destination)) {
transformerFactory.newTransformer().transform(new DOMSource(document), new StreamResult(outputStream));
}
} catch (TransformerException | ParserConfigurationException | SAXException e) {
throw new RuntimeException("Error preprocessing " + inputXml.getAbsolutePath() + ": " + e.getMessage(), e);
}
}
@Override
public ApplicationPackage preprocess(Zone zone, DeployLogger logger) throws IOException {
java.nio.file.Path tempDir = null;
try {
tempDir = Files.createTempDirectory(appDir.getParentFile().toPath(), "preprocess-tempdir");
preprocess(appDir, tempDir.toFile(), zone);
IOUtils.recursiveDeleteDir(preprocessedDir);
// Use 'move' to make sure we do this atomically, important to avoid writing only partial content e.g.
// when shutting down.
// Temp directory needs to be on the same file system as appDir for 'move' to work,
// if it fails (with DirectoryNotEmptyException (!)) we need to use 'copy' instead
// (this will always be the case for the application package for a standalone container).
Files.move(tempDir, preprocessedDir.toPath());
tempDir = null;
} catch (AccessDeniedException | DirectoryNotEmptyException e) {
preprocess(appDir, preprocessedDir, zone);
} finally {
if (tempDir != null)
IOUtils.recursiveDeleteDir(tempDir.toFile());
}
FilesApplicationPackage preprocessedApp = fromFile(preprocessedDir, includeSourceFiles);
preprocessedApp.copyUserDefsIntoApplication();
return preprocessedApp;
}
private void preprocess(File appDir, File dir, Zone zone) throws IOException {
validateServicesFile();
IOUtils.copyDirectory(appDir, dir, - 1,
(__, name) -> ! List.of(preprocessed, SERVICES, HOSTS, CONFIG_DEFINITIONS_DIR).contains(name));
preprocessXML(applicationFile(dir, SERVICES), getServicesFile(), zone);
preprocessXML(applicationFile(dir, HOSTS), getHostsFile(), zone);
}
private void validateServicesFile() throws IOException {
File servicesFile = getServicesFile();
if ( ! servicesFile.exists())
throw new IllegalArgumentException(SERVICES + " does not exist in application package. " +
"There are " + filesInApplicationPackage() + " files in the directory");
if (IOUtils.readFile(servicesFile).isEmpty())
throw new IllegalArgumentException(SERVICES + " in application package is empty. " +
"There are " + filesInApplicationPackage() + " files in the directory");
}
private long filesInApplicationPackage() {
return uncheck(() -> { try (var files = Files.list(appDir.toPath())) { return files.count(); } });
}
private void copyUserDefsIntoApplication() {
File destination = appSubDirs.configDefs();
destination.mkdir();
ConfigDefinitionDir defDir = new ConfigDefinitionDir(destination);
// Copy the user's def files from components.
List bundlesAdded = new ArrayList<>();
for (Component component : getComponents(appSubDirs.root())) {
Bundle bundle = component.getBundle();
defDir.addConfigDefinitionsFromBundle(bundle, bundlesAdded);
bundlesAdded.add(bundle);
}
}
/**
* Computes an md5 hash of the contents of the application package
*
* @return an md5sum of the application package
*/
private static String computeCheckSum(File appDir) {
MessageDigest md5;
try {
md5 = MessageDigest.getInstance("MD5");
for (File file : appDir.listFiles((dir, name) -> !name.equals(EXT_DIR) && !name.startsWith("."))) {
addPathToDigest(file, "", md5, true, false);
}
return toLowerCase(HexDump.toHexString(md5.digest()));
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* Adds the given path to the digest, or does nothing if path is neither file nor dir
*
* @param path path to add to message digest
* @param suffix only files with this suffix are considered
* @param digest the {link @MessageDigest} to add the file paths to
* @param recursive whether to recursively find children in the paths
* @param fullPathNames whether to include the full paths in checksum or only the names
* @throws java.io.IOException if adding path to digest fails when reading files from path
*/
private static void addPathToDigest(File path, String suffix, MessageDigest digest, boolean recursive, boolean fullPathNames) throws IOException {
if (!path.exists()) return;
if (fullPathNames) {
digest.update(path.getPath().getBytes(Utf8.getCharset()));
} else {
digest.update(path.getName().getBytes(Utf8.getCharset()));
}
if (path.isFile()) {
FileInputStream is = new FileInputStream(path);
addToDigest(is, digest);
is.close();
} else if (path.isDirectory()) {
final File[] files = path.listFiles();
if (files != null) {
for (File elem : files) {
if ((elem.isDirectory() && recursive) || elem.getName().endsWith(suffix)) {
addPathToDigest(elem, suffix, digest, recursive, fullPathNames);
}
}
}
}
}
private static final int MD5_BUFFER_SIZE = 65536;
private static void addToDigest(InputStream is, MessageDigest digest) throws IOException {
if (is == null) return;
byte[] buffer = new byte[MD5_BUFFER_SIZE];
int i;
do {
i = is.read(buffer);
if (i > 0) {
digest.update(buffer, 0, i);
}
} while(i != -1);
}
/**
* Builder for {@link com.yahoo.config.model.application.provider.FilesApplicationPackage}. Use
* this to create instances in a flexible manner.
*/
public static class Builder {
private final File appDir;
private Optional preprocessedDir = Optional.empty();
private Optional metaData = Optional.empty();
private boolean includeSourceFiles = false;
public Builder(File appDir) {
this.appDir = appDir;
}
public Builder preprocessedDir(File preprocessedDir) {
this.preprocessedDir = Optional.ofNullable(preprocessedDir);
return this;
}
public Builder deployData(DeployData deployData) {
this.metaData = Optional.of(metaDataFromDeployData(appDir, deployData));
return this;
}
public Builder includeSourceFiles(boolean includeSourceFiles) {
this.includeSourceFiles = includeSourceFiles;
return this;
}
public FilesApplicationPackage build() {
return new FilesApplicationPackage(appDir, preprocessedDir.orElse(applicationFile(appDir, preprocessed)),
metaData.orElse(readMetaData(appDir)), includeSourceFiles);
}
}
static File applicationFile(File parent, String path) {
return applicationFile(parent, Path.fromString(path));
}
static File applicationFile(File parent, Path path) {
File file = new File(parent, path.getRelative());
if ( ! file.getAbsolutePath().startsWith(parent.getAbsolutePath()))
throw new IllegalArgumentException(file + " is not a child of " + parent);
return file;
}
/* Validates that files in application dir and subdirectories have a known extension */
public void validateFileExtensions() {
validFileExtensions.forEach((subDir, __) -> validateInDir(subDir.toFile().toPath()));
}
private void validateInDir(java.nio.file.Path subDir) {
java.nio.file.Path path = appDir.toPath().resolve(subDir);
File subDirectory = path.toFile();
if ( ! subDirectory.exists() || ! subDirectory.isDirectory()) return;
try (var filesInPath = Files.list(path)) {
filesInPath.forEach(filePath -> {
if (filePath.toFile().isDirectory())
validateInDir(appDir.toPath().relativize(filePath));
else
validateFileExtensions(filePath);
});
} catch (IOException e) {
log.log(Level.WARNING, "Unable to list files in " + subDirectory, e);
}
}
static {
// Note: Directories intentionally not validated: MODELS_DIR (custom models can contain files with any extension)
// TODO: Files that according to doc (https://docs.vespa.ai/en/reference/schema-reference.html) can be anywhere in the application package:
// constant tensors (.json, .json.lz4)
// onnx model files (.onnx)
validFileExtensions = Map.ofEntries(
Map.entry(Path.fromString(COMPONENT_DIR), Set.of(".jar")),
Map.entry(CONSTANTS_DIR, Set.of(".json", ".json.lz4")),
Map.entry(Path.fromString(DOCPROCCHAINS_DIR), Set.of(".xml")),
Map.entry(PAGE_TEMPLATES_DIR, Set.of(".xml")),
Map.entry(Path.fromString(PROCESSORCHAINS_DIR), Set.of(".xml")),
Map.entry(QUERY_PROFILES_DIR, Set.of(".xml")),
Map.entry(QUERY_PROFILE_TYPES_DIR, Set.of(".xml")),
Map.entry(Path.fromString(ROUTINGTABLES_DIR), Set.of(".xml")),
Map.entry(RULES_DIR, Set.of(RULES_NAME_SUFFIX)),
// Note: Might have rank profiles in subdirs: [schema-name]/[rank-profile].profile
Map.entry(SCHEMAS_DIR, Set.of(SD_NAME_SUFFIX, RANKEXPRESSION_NAME_SUFFIX, RANKPROFILE_NAME_SUFFIX)),
Map.entry(Path.fromString(SEARCHCHAINS_DIR), Set.of(".xml")),
// Note: Might have rank profiles in subdirs: [schema-name]/[rank-profile].profile
Map.entry(SEARCH_DEFINITIONS_DIR, Set.of(SD_NAME_SUFFIX, RANKEXPRESSION_NAME_SUFFIX, RANKPROFILE_NAME_SUFFIX)),
Map.entry(SECURITY_DIR, Set.of(".pem")));
}
private void validateFileExtensions(java.nio.file.Path pathToFile) {
Set allowedExtensions = findAllowedExtensions(appDir.toPath().relativize(pathToFile).getParent());
log.log(Level.FINE, "Checking " + pathToFile + " against " + allowedExtensions);
String fileName = pathToFile.toFile().getName();
if (allowedExtensions.stream().noneMatch(fileName::endsWith)) {
String message = "File in application package with unknown extension: " +
appDir.toPath().relativize(pathToFile.getParent()).resolve(fileName) + ", please delete or move file to another directory.";
throw new IllegalArgumentException(message);
}
}
private Set findAllowedExtensions(java.nio.file.Path relativeDirectory) {
Set validExtensions = new HashSet<>();
validExtensions.add(".gitignore");
// Special case, since subdirs in schemas/ can have any name
if (isSchemasSubDir(relativeDirectory))
validExtensions.add(RANKPROFILE_NAME_SUFFIX);
else
validExtensions.addAll(validFileExtensions.entrySet().stream()
.filter(entry -> entry.getKey()
.equals(Path.fromString(relativeDirectory.toString())))
.map(Map.Entry::getValue)
.findFirst()
.orElse(Set.of()));
return validExtensions;
}
private boolean isSchemasSubDir(java.nio.file.Path relativeDirectory) {
java.nio.file.Path schemasPath = SCHEMAS_DIR.toFile().toPath().getName(0);
java.nio.file.Path searchDefinitionsPath = SEARCH_DEFINITIONS_DIR.toFile().toPath().getName(0);
if (List.of(schemasPath, searchDefinitionsPath).contains(relativeDirectory)) return false;
return (relativeDirectory.startsWith(schemasPath + "/")
|| relativeDirectory.startsWith(searchDefinitionsPath + "/"));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy