org.neo4j.configuration.helpers.FromPaths Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of neo4j-configuration Show documentation
Show all versions of neo4j-configuration Show documentation
Code responsible for parsing the Neo4j configuration.
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [http://neo4j.com]
*
* This file is part of Neo4j.
*
* Neo4j is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.neo4j.configuration.helpers;
import org.apache.commons.lang3.StringUtils;
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Arrays;
import java.util.Objects;
import java.util.Set;
import java.util.stream.Collectors;
import org.neo4j.io.fs.DefaultFileSystemAbstraction;
import org.neo4j.io.fs.FileSystemAbstraction;
import static java.util.stream.Collectors.joining;
import static org.neo4j.io.IOUtils.uncheckedFunction;
public class FromPaths
{
private final Set paths;
private final FileSystemAbstraction fs = new DefaultFileSystemAbstraction();
public FromPaths( String value )
{
validateNotEmpty( value );
this.paths = buildPaths( value );
}
public boolean isSingle()
{
return paths.size() == 1;
}
public Set getPaths()
{
return paths;
}
private Set getAllFolders( Path path ) throws IOException
{
return Arrays.stream( fs.listFiles( path ) )
.filter( Files::isDirectory )
.collect( Collectors.toSet() );
}
private Set buildPaths( String value )
{
final var tokens = value.split( "," );
return Arrays.stream( tokens )
.map( String::trim )
.filter( t -> !t.isEmpty() )
.map( path -> new File( path ).getAbsoluteFile() ) // Path class can't be used because Path can't be created with regex for some
// file systems
.peek( file ->
{
validateParentPath( file ); //Path class can't contain regex in the subpath
validateLastSubPath( file );
} )
.flatMap( uncheckedFunction( file -> getAndFilterPaths( file ).stream() ) )
.collect( Collectors.toSet() );
}
private Set getAndFilterPaths( File file ) throws IOException
{
final var parent = file.getParent(); //not null, protect by validateParentPath
final var pattern = new DatabaseNamePattern( file.getName() );
if ( !pattern.containsPattern() )
{
return Set.of( Path.of( file.toString() ) );
}
return getAllFolders( Path.of( parent ) ).stream()
.filter( path ->
{
final var name = path.getName( path.getNameCount() - 1 );
return pattern.matches( name.toString() );
} )
.collect( Collectors.toSet() );
}
private static void validateParentPath( File file )
{
final var parentPath = file.getParent();
if ( parentPath != null && !parentPath.trim().isEmpty() )
{
final var asterisks = StringUtils.countMatches( parentPath, '*' );
final var questionMarks = StringUtils.countMatches( parentPath, '?' );
if ( asterisks > 0 || questionMarks > 0 )
{
throw new IllegalArgumentException( file.getAbsolutePath() + " is illegal. Asterisks and question marks should be placed in the last subpath" );
}
}
else
{
throw new IllegalArgumentException( "From path with value=" + file.getAbsolutePath() + " should not point to the root of the file system" );
}
}
private static void validateLastSubPath( File file )
{
if ( file.getParent() == null || Path.of( file.getParent() ).getNameCount() == 0 )
{
throw new IllegalArgumentException( "From path with value=" + file.getAbsolutePath() + " should not point to the root of the file system" );
}
final var lastSubPath = file.getName();
try
{
new DatabaseNamePattern( lastSubPath );
}
catch ( IllegalArgumentException ex )
{
throw new IllegalArgumentException( "Last path of " + file.getAbsolutePath() + " is in illegal format.", ex );
}
}
private static void validateNotEmpty( String path )
{
Objects.requireNonNull( path, "The provided from parameter is empty." );
if ( path.trim().isEmpty() )
{
throw new IllegalArgumentException( "The provided from parameter is empty." );
}
}
@Override
public String toString()
{
return paths == null ? "" : paths.stream()
.filter( Objects::nonNull )
.map( Path::toAbsolutePath )
.map( Object::toString )
.collect( joining( ", " ) );
}
}