All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.neo4j.kernel.StandardExpander Maven / Gradle / Ivy

Go to download

Neo4j kernel is a lightweight, embedded Java database designed to store data structured as graphs rather than tables. For more information, see http://neo4j.org.

There is a newer version: 5.26.0
Show newest version
/*
 * Copyright (c) 2002-2015 "Neo Technology,"
 * Network Engine for Objects in Lund AB [http://neotechnology.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.kernel;

import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.EnumMap;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import org.neo4j.graphdb.Direction;
import org.neo4j.graphdb.Expander;
import org.neo4j.graphdb.Node;
import org.neo4j.graphdb.NotFoundException;
import org.neo4j.graphdb.Path;
import org.neo4j.graphdb.PathExpander;
import org.neo4j.graphdb.Relationship;
import org.neo4j.graphdb.RelationshipExpander;
import org.neo4j.graphdb.RelationshipType;
import org.neo4j.graphdb.traversal.BranchState;
import org.neo4j.helpers.Pair;
import org.neo4j.helpers.Predicate;
import org.neo4j.helpers.collection.ArrayIterator;
import org.neo4j.helpers.collection.FilteringIterator;
import org.neo4j.helpers.collection.IteratorWrapper;
import org.neo4j.helpers.collection.NestingIterator;
import org.neo4j.kernel.impl.util.SingleNodePath;

import static java.util.Arrays.asList;
import static org.neo4j.kernel.ExtendedPath.extend;

public abstract class StandardExpander implements Expander, PathExpander
{
    private StandardExpander()
    {
    }

    static abstract class StandardExpansion implements Expansion
    {
        final StandardExpander expander;
        final Path path;
        final BranchState state;

        StandardExpansion( StandardExpander expander, Path path, BranchState state )
        {
            this.expander = expander;
            this.path = path;
            this.state = state;
        }

        String stringRepresentation( String nodesORrelationships )
        {
            return "Expansion[" + path + ".expand( " + expander + " )."
                    + nodesORrelationships + "()]";
        }

        abstract StandardExpansion createNew( StandardExpander expander );

        public StandardExpansion including( RelationshipType type )
        {
            return createNew( expander.add( type ) );
        }

        public StandardExpansion including( RelationshipType type,
                                               Direction direction )
        {
            return createNew( expander.add( type, direction ) );
        }

        public StandardExpansion excluding( RelationshipType type )
        {
            return createNew( expander.remove( type ) );
        }

        public StandardExpander expander()
        {
            return expander;
        }

        public StandardExpansion filterNodes( Predicate filter )
        {
            return createNew( expander.addNodeFilter( filter ) );
        }

        public StandardExpansion filterRelationships(
                Predicate filter )
        {
            return createNew( expander.addRelationshipFilter( filter ) );
        }

        public T getSingle()
        {
            final Iterator expanded = iterator();
            if ( expanded.hasNext() )
            {
                final T result = expanded.next();
                if ( expanded.hasNext() )
                {
                    throw new NotFoundException(
                            "More than one relationship found for " + this );
                }
                return result;
            }
            return null;
        }

        public boolean isEmpty()
        {
            return !expander.doExpand( path, state ).hasNext();
        }

        public StandardExpansion nodes()
        {
            return new NodeExpansion( expander, path, state );
        }

        public StandardExpansion relationships()
        {
            return new RelationshipExpansion( expander, path, state );
        }

        public StandardExpansion> pairs()
        {
            return new PairExpansion( expander, path, state );
        }
    }

    private static final class RelationshipExpansion extends
            StandardExpansion
    {
        RelationshipExpansion( StandardExpander expander, Path path, BranchState state )
        {
            super( expander, path, state );
        }

        @Override
        public String toString()
        {
            return stringRepresentation( "relationships" );
        }

        @Override
        StandardExpansion createNew( StandardExpander expander )
        {
            return new RelationshipExpansion( expander, path, state );
        }

        @Override
        public StandardExpansion relationships()
        {
            return this;
        }

        public Iterator iterator()
        {
            return expander.doExpand( path, state );
        }
    }

    private static final class NodeExpansion extends StandardExpansion
    {
        NodeExpansion( StandardExpander expander, Path path, BranchState state )
        {
            super( expander, path, state );
        }

        @Override
        public String toString()
        {
            return stringRepresentation( "nodes" );
        }

        @Override
        StandardExpansion createNew( StandardExpander expander )
        {
            return new NodeExpansion( expander, path, state );
        }

        @Override
        public StandardExpansion nodes()
        {
            return this;
        }

        public Iterator iterator()
        {
            final Node node = path.endNode();
            return new IteratorWrapper(
                    expander.doExpand( path, state ) )
            {
                @Override
                protected Node underlyingObjectToObject( Relationship rel )
                {
                    return rel.getOtherNode( node );
                }
            };
        }
    }

    private static final class PairExpansion extends
            StandardExpansion>
    {
        PairExpansion( StandardExpander expander, Path path, BranchState state )
        {
            super( expander, path, state );
        }

        @Override
        public String toString()
        {
            return stringRepresentation( "pairs" );
        }

        @Override
        StandardExpansion> createNew( StandardExpander expander )
        {
            return new PairExpansion( expander, path, state );
        }

        @Override
        public StandardExpansion> pairs()
        {
            return this;
        }

        public Iterator> iterator()
        {
            final Node node = path.endNode();
            return new IteratorWrapper, Relationship>(
                    expander.doExpand( path, state ) )
            {
                @Override
                protected Pair underlyingObjectToObject(
                        Relationship rel )
                {
                    return Pair.of( rel, rel.getOtherNode( node ) );
                }
            };
        }
    }

    private static class AllExpander extends StandardExpander
    {
        private final Direction direction;

        AllExpander( Direction direction )
        {
            this.direction = direction;
        }

        @Override
        void buildString( StringBuilder result )
        {
            if ( direction != Direction.BOTH )
            {
                result.append( direction );
                result.append( ":" );
            }
            result.append( "*" );
        }

        @Override
        Iterator doExpand( Path path, BranchState state )
        {
            return path.endNode().getRelationships( direction ).iterator();
        }

        @Override
        public StandardExpander add( RelationshipType type, Direction dir )
        {
            return this;
        }

        @Override
        public StandardExpander remove( RelationshipType type )
        {
            Map exclude = new HashMap();
            exclude.put( type.name(), Exclusion.ALL );
            return new ExcludingExpander( Exclusion.include( direction ),
                    exclude );
        }

        @Override
        public StandardExpander reversed()
        {
            return reverse();
        }

        @Override
        public StandardExpander reverse()
        {
            return new AllExpander( direction.reverse() );
        }
    }

    private enum Exclusion
    {
        ALL( null, "!" )
                {
                    @Override
                    public boolean accept( Node start, Relationship rel )
                    {
                        return false;
                    }
                },
        INCOMING( Direction.OUTGOING )
                {
                    @Override
                    Exclusion reversed()
                    {
                        return OUTGOING;
                    }
                },
        OUTGOING( Direction.INCOMING )
                {
                    @Override
                    Exclusion reversed()
                    {
                        return INCOMING;
                    }
                },
        NONE( Direction.BOTH, "" )
                {
                    @Override
                    boolean includes( Direction direction )
                    {
                        return true;
                    }
                };
        private final String string;
        private final Direction direction;

        private Exclusion( Direction direction, String string )
        {
            this.direction = direction;
            this.string = string;
        }

        private Exclusion( Direction direction )
        {
            this.direction = direction;
            this.string = "!" + name() + ":";
        }

        @Override
        public final String toString()
        {
            return string;
        }

        boolean accept( Node start, Relationship rel )
        {
            return matchDirection( direction, start, rel );
        }

        Exclusion reversed()
        {
            return this;
        }

        boolean includes( Direction dir )
        {
            return this.direction == dir;
        }

        static Exclusion include( Direction direction )
        {
            switch ( direction )
            {
                case INCOMING:
                    return OUTGOING;
                case OUTGOING:
                    return INCOMING;
                default:
                    return NONE;
            }
        }
    }

    private static final class ExcludingExpander extends StandardExpander
    {
        private final Exclusion defaultExclusion;
        private final Map exclusion;

        ExcludingExpander( Exclusion defaultExclusion,
                           Map exclusion )
        {
            this.defaultExclusion = defaultExclusion;
            this.exclusion = exclusion;
        }

        @Override
        void buildString( StringBuilder result )
        {
            // FIXME: not really correct
            result.append( defaultExclusion );
            result.append( "*" );
            for ( Map.Entry entry : exclusion.entrySet() )
            {
                result.append( "," );
                result.append( entry.getValue() );
                result.append( entry.getKey() );
            }
        }

        @Override
        Iterator doExpand( Path path, BranchState state )
        {
            final Node node = path.endNode();
            return new FilteringIterator(
                    node.getRelationships().iterator(),
                    new Predicate()
                    {
                        public boolean accept( Relationship rel )
                        {
                            Exclusion exclude = exclusion.get( rel.getType().name() );
                            exclude = (exclude == null) ? defaultExclusion
                                    : exclude;
                            return exclude.accept( node, rel );
                        }
                    } );
        }

        @Override
        public StandardExpander add( RelationshipType type, Direction direction )
        {
            Exclusion excluded = exclusion.get( type.name() );
            final Map newExclusion;
            if ( (excluded == null ? defaultExclusion : excluded).includes( direction ) )
            {
                return this;
            }
            else
            {
                excluded = Exclusion.include( direction );
                if ( excluded == defaultExclusion )
                {
                    if ( exclusion.size() == 1 )
                    {
                        return new AllExpander( defaultExclusion.direction );
                    }
                    else
                    {
                        newExclusion = new HashMap(
                                exclusion );
                        newExclusion.remove( type.name() );
                    }
                }
                else
                {
                    newExclusion = new HashMap( exclusion );
                    newExclusion.put( type.name(), excluded );
                }
            }
            return new ExcludingExpander( defaultExclusion, newExclusion );
        }

        @Override
        public StandardExpander remove( RelationshipType type )
        {
            Exclusion excluded = exclusion.get( type.name() );
            if ( excluded == Exclusion.ALL )
            {
                return this;
            }
            Map newExclusion = new HashMap(
                    exclusion );
            newExclusion.put( type.name(), Exclusion.ALL );
            return new ExcludingExpander( defaultExclusion, newExclusion );
        }

        @Override
        public StandardExpander reversed()
        {
            return reverse();
        }

        @Override
        public StandardExpander reverse()
        {
            Map newExclusion = new HashMap();
            for ( Map.Entry entry : exclusion.entrySet() )
            {
                newExclusion.put( entry.getKey(), entry.getValue().reversed() );
            }
            return new ExcludingExpander( defaultExclusion.reversed(), newExclusion );
        }
    }

    public static final StandardExpander DEFAULT = new AllExpander(
            Direction.BOTH )
    {
        @Override
        public StandardExpander add( RelationshipType type, Direction direction )
        {
            return create( type, direction );
        }
    };

    public static final StandardExpander EMPTY =
            new RegularExpander( Collections.emptyMap() );

    private static class DirectionAndTypes
    {
        final Direction direction;
        final RelationshipType[] types;

        DirectionAndTypes( Direction direction, RelationshipType[] types )
        {
            this.direction = direction;
            this.types = types;
        }
    }

    static class RegularExpander extends StandardExpander
    {
        final Map typesMap;
        final DirectionAndTypes[] directions;

        RegularExpander( Map types )
        {
            this.typesMap = types;
            this.directions = new DirectionAndTypes[types.size()];
            int i = 0;
            for ( Map.Entry entry : types.entrySet() )
            {
                this.directions[i++] = new DirectionAndTypes( entry.getKey(), entry.getValue() );
            }
        }

        @Override
        void buildString( StringBuilder result )
        {
            result.append( typesMap.toString() );
        }

        @Override
        Iterator doExpand( Path path, BranchState state )
        {
            final Node node = path.endNode();
            if ( directions.length == 1 )
            {
                DirectionAndTypes direction = directions[0];
                return node.getRelationships( direction.direction, direction.types ).iterator();
            }
            else
            {
                return new NestingIterator( new ArrayIterator(
                        directions ) )
                {
                    @Override
                    protected Iterator createNestedIterator( DirectionAndTypes item )
                    {
                        return node.getRelationships( item.direction, item.types ).iterator();
                    }
                };
            }
        }

        StandardExpander createNew( Map types )
        {
            if ( types.isEmpty() )
            {
                return new AllExpander( Direction.BOTH );
            }
            return new RegularExpander( types );
        }

        @Override
        public StandardExpander add( RelationshipType type, Direction direction )
        {
            Map> tempMap = temporaryTypeMapFrom( typesMap );
            tempMap.get( direction ).add( type );
            return createNew( toTypeMap( tempMap ) );
        }

        @Override
        public StandardExpander remove( RelationshipType type )
        {
            Map> tempMap = temporaryTypeMapFrom( typesMap );
            for ( Direction direction : Direction.values() )
            {
                tempMap.get( direction ).remove( type );
            }
            return createNew( toTypeMap( tempMap ) );
        }

        @Override
        public StandardExpander reversed()
        {
            return reverse();
        }

        @Override
        public StandardExpander reverse()
        {
            Map> tempMap = temporaryTypeMapFrom( typesMap );
            Collection out = tempMap.get( Direction.OUTGOING );
            Collection in = tempMap.get( Direction.INCOMING );
            tempMap.put( Direction.OUTGOING, in );
            tempMap.put( Direction.INCOMING, out );
            return createNew( toTypeMap( tempMap ) );
        }
    }

    private static final class FilteringExpander extends StandardExpander
    {
        private final StandardExpander expander;
        private final Filter[] filters;

        FilteringExpander( StandardExpander expander, Filter... filters )
        {
            this.expander = expander;
            this.filters = filters;
        }

        @Override
        void buildString( StringBuilder result )
        {
            expander.buildString( result );
            result.append( "; filter:" );
            for ( Filter filter : filters )
            {
                result.append( " " );
                result.append( filter );
            }
        }

        @Override
        Iterator doExpand( final Path path, BranchState state )
        {
            return new FilteringIterator(
                    expander.doExpand( path, state ), new Predicate()
            {
                public boolean accept( Relationship item )
                {
                    Path extendedPath = extend( path, item );
                    for ( Filter filter : filters )
                    {
                        if ( filter.exclude( extendedPath ) )
                        {
                            return false;
                        }
                    }
                    return true;
                }
            } );
        }

        @Override
        public StandardExpander addNodeFilter( Predicate filter )
        {
            return new FilteringExpander( expander, append( filters,
                    new NodeFilter( filter ) ) );
        }

        @Override
        public StandardExpander addRelationshipFilter(
                Predicate filter )
        {
            return new FilteringExpander( expander, append( filters,
                    new RelationshipFilter( filter ) ) );
        }

        @Override
        public StandardExpander add( RelationshipType type, Direction direction )
        {
            return new FilteringExpander( expander.add( type, direction ),
                    filters );
        }

        @Override
        public StandardExpander remove( RelationshipType type )
        {
            return new FilteringExpander( expander.remove( type ), filters );
        }

        @Override
        public StandardExpander reversed()
        {
            return reverse();
        }

        @Override
        public StandardExpander reverse()
        {
            return new FilteringExpander( expander.reversed(), filters );
        }
    }

    private static final class WrappingExpander extends StandardExpander
    {
        private static final String IMMUTABLE = "Immutable Expander ";
        private final PathExpander expander;

        WrappingExpander( PathExpander expander )
        {
            this.expander = expander;
        }

        @Override
        void buildString( StringBuilder result )
        {
            result.append( expander );
        }

        @Override
        Iterator doExpand( Path path, BranchState state )
        {
            return expander.expand( path, state ).iterator();
        }

        @Override
        public StandardExpander add( RelationshipType type, Direction direction )
        {
            throw new UnsupportedOperationException( IMMUTABLE + expander );
        }

        @Override
        public StandardExpander remove( RelationshipType type )
        {
            throw new UnsupportedOperationException( IMMUTABLE + expander );
        }

        @Override
        public StandardExpander reversed()
        {
            return reverse();
        }

        @Override
        public StandardExpander reverse()
        {
            throw new UnsupportedOperationException( IMMUTABLE + expander );
        }
    }

    private static final class WrappingRelationshipExpander extends StandardExpander
    {
        private static final String IMMUTABLE = "Immutable Expander ";
        private final RelationshipExpander expander;

        WrappingRelationshipExpander( RelationshipExpander expander )
        {
            this.expander = expander;
        }

        @Override
        void buildString( StringBuilder result )
        {
            result.append( expander );
        }

        @Override
        Iterator doExpand( Path path, BranchState state )
        {
            return expander.expand( path.endNode() ).iterator();
        }

        @Override
        public StandardExpander add( RelationshipType type, Direction direction )
        {
            throw new UnsupportedOperationException( IMMUTABLE + expander );
        }

        @Override
        public StandardExpander remove( RelationshipType type )
        {
            throw new UnsupportedOperationException( IMMUTABLE + expander );
        }

        @Override
        public StandardExpander reversed()
        {
            return reverse();
        }

        @Override
        public StandardExpander reverse()
        {
            throw new UnsupportedOperationException( IMMUTABLE + expander );
        }
    }

    private static abstract class Filter
    {
        abstract boolean exclude( Path path );
    }

    private static final class NodeFilter extends Filter
    {
        private final Predicate predicate;

        NodeFilter( Predicate predicate )
        {
            this.predicate = predicate;
        }

        @Override
        public String toString()
        {
            return predicate.toString();
        }

        @Override
        boolean exclude( Path path )
        {
            return !predicate.accept( path.lastRelationship().getOtherNode( path.endNode() ) );
        }
    }

    private static final class RelationshipFilter extends Filter
    {
        private final Predicate predicate;

        RelationshipFilter( Predicate predicate )
        {
            this.predicate = predicate;
        }

        @Override
        public String toString()
        {
            return predicate.toString();
        }

        @Override
        boolean exclude( Path path )
        {
            return !predicate.accept( path.lastRelationship() );
        }
    }

    private static final class PathFilter extends Filter
    {
        private final Predicate predicate;

        PathFilter( Predicate predicate )
        {
            this.predicate = predicate;
        }

        @Override
        public String toString()
        {
            return predicate.toString();
        }

        @Override
        boolean exclude( Path path )
        {
            return !predicate.accept( path );
        }
    }

    public final Expansion expand( Node node )
    {
        return new RelationshipExpansion( this, new SingleNodePath( node ), BranchState.NO_STATE );
    }

    public final Expansion expand( Path path, BranchState state )
    {
        return new RelationshipExpansion( this, path, state );
    }

    static  T[] append( T[] array, T item )
    {
        @SuppressWarnings("unchecked") T[] result = (T[]) Array.newInstance(
                array.getClass().getComponentType(), array.length + 1 );
        System.arraycopy( array, 0, result, 0, array.length );
        result[array.length] = item;
        return result;
    }

    static boolean matchDirection( Direction dir, Node start, Relationship rel )
    {
        switch ( dir )
        {
            case INCOMING:
                return rel.getEndNode().equals( start );
            case OUTGOING:
                return rel.getStartNode().equals( start );
            case BOTH:
                return true;
        }
        return true;
    }

    abstract Iterator doExpand( Path path, BranchState state );

    @Override
    public final String toString()
    {
        StringBuilder result = new StringBuilder( "Expander[" );
        buildString( result );
        result.append( "]" );
        return result.toString();
    }

    abstract void buildString( StringBuilder result );

    public final StandardExpander add( RelationshipType type )
    {
        return add( type, Direction.BOTH );
    }

    public abstract StandardExpander add( RelationshipType type,
                                          Direction direction );

    public abstract StandardExpander remove( RelationshipType type );

    public abstract StandardExpander reverse();

    public abstract StandardExpander reversed();

    public StandardExpander addNodeFilter( Predicate filter )
    {
        return new FilteringExpander( this, new NodeFilter( filter ) );
    }

    @Override
    public final Expander addRelationsipFilter( Predicate filter )
    {
        return addRelationshipFilter( filter );
    }

    public StandardExpander addRelationshipFilter(
            Predicate filter )
    {
        return new FilteringExpander( this, new RelationshipFilter( filter ) );
    }

    static StandardExpander wrap( RelationshipExpander expander )
    {
        return new WrappingRelationshipExpander( expander );
    }

    static StandardExpander wrap( PathExpander expander )
    {
        return new WrappingExpander( expander );
    }

    public static PathExpander toPathExpander( RelationshipExpander expander )
    {
        return expander instanceof PathExpander ? (PathExpander) expander : wrap( expander );
    }

    public static StandardExpander create( Direction direction )
    {
        return new AllExpander( direction );
    }

    public static StandardExpander create( RelationshipType type, Direction dir )
    {
        Map types =
                new EnumMap( Direction.class );
        types.put( dir, new RelationshipType[]{type} );
        return new RegularExpander( types );
    }

    static StandardExpander create( RelationshipType type1, Direction dir1,
                                    RelationshipType type2, Direction dir2 )
    {
        Map> tempMap = temporaryTypeMap();
        tempMap.get( dir1 ).add( type1 );
        tempMap.get( dir2 ).add( type2 );
        return new RegularExpander( toTypeMap( tempMap ) );
    }

    private static Map toTypeMap(
            Map> tempMap )
    {
        // Remove OUT/IN where there is a BOTH
        Collection both = tempMap.get( Direction.BOTH );
        tempMap.get( Direction.OUTGOING ).removeAll( both );
        tempMap.get( Direction.INCOMING ).removeAll( both );

        // Convert into a final map
        Map map = new EnumMap( Direction.class );
        for ( Map.Entry> entry : tempMap.entrySet() )
        {
            if ( !entry.getValue().isEmpty() )
            {
                map.put( entry.getKey(), entry.getValue().toArray( new RelationshipType[entry.getValue().size()] ) );
            }
        }
        return map;
    }

    private static Map> temporaryTypeMap()
    {
        Map> map = new EnumMap>( Direction.class );
        for ( Direction direction : Direction.values() )
        {
            map.put( direction, new ArrayList() );
        }
        return map;
    }

    private static Map> temporaryTypeMapFrom( Map typeMap )
    {
        Map> map = new EnumMap>( Direction.class );
        for ( Direction direction : Direction.values() )
        {
            ArrayList types = new ArrayList();
            map.put( direction, types );
            RelationshipType[] existing = typeMap.get( direction );
            if ( existing != null )
            {
                types.addAll( asList( existing ) );
            }
        }
        return map;
    }

    public static StandardExpander create( RelationshipType type1, Direction dir1,
                                           RelationshipType type2, Direction dir2, Object... more )
    {
        Map> tempMap = temporaryTypeMap();
        tempMap.get( dir1 ).add( type1 );
        tempMap.get( dir2 ).add( type2 );
        for ( int i = 0; i < more.length; i++ )
        {
            RelationshipType type = (RelationshipType) more[i++];
            Direction direction = (Direction) more[i];
            tempMap.get( direction ).add( type );
        }
        return new RegularExpander( toTypeMap( tempMap ) );
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy