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

org.neo4j.values.virtual.MapValue Maven / Gradle / Ivy

There is a newer version: 5.26.0
Show newest version
/*
 * Copyright (c) 2002-2020 "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.values.virtual;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.function.BiFunction;
import java.util.stream.StreamSupport;

import org.neo4j.function.ThrowingBiConsumer;
import org.neo4j.internal.helpers.collection.PrefetchingIterator;
import org.neo4j.values.AnyValue;
import org.neo4j.values.AnyValueWriter;
import org.neo4j.values.Comparison;
import org.neo4j.values.Equality;
import org.neo4j.values.TernaryComparator;
import org.neo4j.values.ValueMapper;
import org.neo4j.values.VirtualValue;
import org.neo4j.values.storable.Values;

import static org.neo4j.values.storable.Values.NO_VALUE;

public abstract class MapValue extends VirtualValue
{
    public static final MapValue EMPTY = new MapValue()
    {
        @Override
        public Iterable keySet()
        {
            return Collections.emptyList();
        }

        @Override
        public  void foreach( ThrowingBiConsumer f )
        {
            //do nothing
        }

        @Override
        public boolean containsKey( String key )
        {
            return false;
        }

        @Override
        public AnyValue get( String key )
        {
            return NO_VALUE;
        }

        @Override
        public int size()
        {
            return 0;
        }

        @Override
        protected long estimatedPayloadSize()
        {
            return 0L;
        }

        @Override
        public long estimatedHeapUsage()
        {
            //EMPTY is a singleton and doesn't add to heap usage
            return 0L;
        }
    };

    static final class MapWrappingMapValue extends MapValue
    {
        private final Map map;

        MapWrappingMapValue( Map map )
        {
            this.map = map;
        }

        @Override
        public Iterable keySet()
        {
            return map.keySet();
        }

        @Override
        public  void foreach( ThrowingBiConsumer f ) throws E
        {
            for ( Map.Entry entry : map.entrySet() )
            {
                f.accept( entry.getKey(), entry.getValue() );
            }
        }

        @Override
        public boolean containsKey( String key )
        {
            return map.containsKey( key );
        }

        @Override
        public AnyValue get( String key )
        {
            return map.getOrDefault( key, NO_VALUE );
        }

        @Override
        public int size()
        {
            return map.size();
        }
    }

    private static final class FilteringMapValue extends MapValue
    {
        private final MapValue map;
        private final BiFunction filter;
        private int size = -1;

        FilteringMapValue( MapValue map,
                BiFunction filter )
        {
            this.map = map;
            this.filter = filter;
        }

        @Override
        public Iterable keySet()
        {
            List keys = size >= 0 ? new ArrayList<>( size ) : new ArrayList<>();
            foreach( ( key, value ) -> {
                if ( filter.apply( key, value ) )
                {
                    keys.add( key );
                }
            } );

            return keys;
        }

        @Override
        public  void foreach( ThrowingBiConsumer f ) throws E
        {
            map.foreach( ( s, anyValue ) -> {
                if ( filter.apply( s, anyValue ) )
                {
                    f.accept( s, anyValue );
                }
            } );
        }

        @Override
        public boolean containsKey( String key )
        {
            AnyValue value = map.get( key );
            if ( value == NO_VALUE )
            {
                return false;
            }
            else
            {
                return filter.apply( key, value );
            }
        }

        @Override
        public AnyValue get( String key )
        {
            AnyValue value = map.get( key );
            if ( value == NO_VALUE )
            {
                return NO_VALUE;
            }
            else if ( filter.apply( key, value ) )
            {
                return value;
            }
            else
            {
                return NO_VALUE;
            }
        }

        @Override
        public int size()
        {
            if ( size < 0 )
            {
                size = 0;
                foreach( ( k, v ) -> {
                    if ( filter.apply( k, v ) )
                    {
                        size++;
                    }
                } );
            }
            return size;
        }
    }

    private static final class MappedMapValue extends MapValue
    {
        private final MapValue map;
        private final BiFunction mapFunction;

        MappedMapValue( MapValue map,
                BiFunction mapFunction )
        {
            this.map = map;
            this.mapFunction = mapFunction;
        }

        @Override
        public ListValue keys()
        {
            return map.keys();
        }

        @Override
        public Iterable keySet()
        {
            return map.keySet();
        }

        @Override
        public  void foreach( ThrowingBiConsumer f ) throws E
        {
            map.foreach( ( s, anyValue ) -> f.accept( s, mapFunction.apply( s, anyValue ) ) );
        }

        @Override
        public boolean containsKey( String key )
        {
            return map.containsKey( key );
        }

        @Override
        public AnyValue get( String key )
        {
            return mapFunction.apply( key, map.get( key ) );
        }

        @Override
        public int size()
        {
            return map.size();
        }
    }

    private static final class UpdatedMapValue extends MapValue
    {
        private final MapValue map;
        private final String[] updatedKeys;
        private final AnyValue[] updatedValues;

        UpdatedMapValue( MapValue map, String[] updatedKeys, AnyValue[] updatedValues )
        {
            assert updatedKeys.length == updatedValues.length;
            assert !overlaps( map, updatedKeys );
            this.map = map;
            this.updatedKeys = updatedKeys;
            this.updatedValues = updatedValues;
        }

        private static boolean overlaps( MapValue map, String[] updatedKeys )
        {
            for ( String key : updatedKeys )
            {
                if ( map.containsKey( key ) )
                {
                    return true;
                }
            }

            return false;
        }

        @Override
        public ListValue keys()
        {
            return VirtualValues.concat( map.keys(), VirtualValues.fromArray( Values.stringArray( updatedKeys ) ) );
        }

        @Override
        public Iterable keySet()
        {
            return () -> new Iterator<>()
            {
                private Iterator internal = map.keySet().iterator();
                private int index;

                @Override
                public boolean hasNext()
                {
                    if ( internal.hasNext() )
                    {
                        return true;
                    }
                    else
                    {
                        return index < updatedKeys.length;
                    }
                }

                @Override
                public String next()
                {
                    if ( internal.hasNext() )
                    {
                        return internal.next();
                    }
                    else if ( index < updatedKeys.length )
                    {
                        return updatedKeys[index++];
                    }
                    else
                    {
                        throw new NoSuchElementException();
                    }
                }
            };
        }

        @Override
        public  void foreach( ThrowingBiConsumer f ) throws E
        {
            map.foreach( f );
            for ( int i = 0; i < updatedKeys.length; i++ )
            {
                f.accept( updatedKeys[i], updatedValues[i] );
            }
        }

        @Override
        public boolean containsKey( String key )
        {
            for ( String updatedKey : updatedKeys )
            {
                if ( updatedKey.equals( key ) )
                {
                    return true;
                }
            }

            return map.containsKey( key );
        }

        @Override
        public AnyValue get( String key )
        {
            for ( int i = 0; i < updatedKeys.length; i++ )
            {
                if ( updatedKeys[i].equals( key ) )
                {
                    return updatedValues[i];
                }
            }
            return map.get( key );
        }

        @Override
        public int size()
        {
            return map.size() + updatedKeys.length;
        }
    }

    private static final class CombinedMapValue extends MapValue
    {
        private final MapValue[] maps;

        CombinedMapValue( MapValue... mapValues )
        {
            this.maps = mapValues;
        }

        @Override
        public Iterable keySet()
        {
           return () -> new PrefetchingIterator<>()
           {
               private int mapIndex;
               private Iterator internal;
               private HashSet seen = new HashSet<>();

               @Override
               protected String fetchNextOrNull()
               {
                   while ( mapIndex < maps.length || internal != null && internal.hasNext() )
                   {
                       if ( internal == null || !internal.hasNext() )
                       {
                           internal = maps[mapIndex++].keySet().iterator();
                       }

                       while ( internal.hasNext() )
                       {
                           String key = internal.next();
                           if ( seen.add( key ) )
                           {
                               return key;
                           }
                       }
                   }
                   return null;
               }
           };
        }

        @Override
        public  void foreach( ThrowingBiConsumer f ) throws E
        {
            HashSet seen = new HashSet<>();
            ThrowingBiConsumer consume = ( key, value ) ->
            {
                if ( seen.add( key ) )
                {
                    f.accept( key, value );
                }
            };
            for ( int i = maps.length - 1; i >= 0; i-- )
            {
                maps[i].foreach( consume );
            }
        }

        @Override
        public boolean containsKey( String key )
        {
            for ( MapValue map : maps )
            {
                if ( map.containsKey( key ) )
                {
                    return true;
                }
            }
            return false;
        }

        @Override
        public AnyValue get( String key )
        {
            for ( int i = maps.length - 1; i >= 0; i-- )
            {
                AnyValue value = maps[i].get( key );
                if ( value != NO_VALUE )
                {
                    return value;
                }
            }
            return NO_VALUE;
        }

        @Override
        public int size()
        {
            int[] size = {0};
            HashSet seen = new HashSet<>();
            ThrowingBiConsumer consume = ( key, value ) ->
            {
                if ( seen.add( key ) )
                {
                    size[0]++;
                }
            };
            for ( int i = maps.length - 1; i >= 0; i-- )
            {
                maps[i].foreach( consume );
            }
            return size[0];
        }
    }

    @Override
    public int computeHash()
    {
        int[] h = new int[1];
        foreach( ( key, value ) -> h[0] += key.hashCode() ^ value.hashCode() );
        return h[0];
    }

    @Override
    public  void writeTo( AnyValueWriter writer ) throws E
    {
        writer.beginMap( size() );
        foreach( ( s, anyValue ) -> {
            writer.writeString( s );
            anyValue.writeTo( writer );
        } );
        writer.endMap();
    }

    @Override
    public boolean equals( VirtualValue other )
    {
        if ( !(other instanceof MapValue) )
        {
            return false;
        }
        MapValue that = (MapValue) other;
        int size = size();
        if ( size != that.size() )
        {
            return false;
        }

        Iterable keys = keySet();
        for ( String key : keys )
        {
            if ( !get( key ).equals( that.get( key ) ) )
            {
                return false;
            }
        }

        return true;
    }

    public abstract Iterable keySet();

    public ListValue keys()
    {
        String[] keys = new String[size()];
        int i = 0;
        for ( String key : keySet() )
        {
            keys[i++] = key;
        }
        return VirtualValues.fromArray( Values.stringArray( keys ) );
    }

    @Override
    public VirtualValueGroup valueGroup()
    {
        return VirtualValueGroup.MAP;
    }

    @Override
    public int unsafeCompareTo( VirtualValue other, Comparator comparator )
    {
        MapValue otherMap = (MapValue) other;
        int size = size();
        int compare = Integer.compare( size, otherMap.size() );
        if ( compare == 0 )
        {
            String[] thisKeys = StreamSupport.stream( keySet().spliterator(), false).toArray( String[]::new  );
            Arrays.sort( thisKeys, String::compareTo );
            String[] thatKeys = StreamSupport.stream( otherMap.keySet().spliterator(), false).toArray( String[]::new  );
            Arrays.sort( thatKeys, String::compareTo );
            for ( int i = 0; i < size; i++ )
            {
                compare = thisKeys[i].compareTo( thatKeys[i] );
                if ( compare != 0 )
                {
                    return compare;
                }
            }

            for ( int i = 0; i < size; i++ )
            {
                String key = thisKeys[i];
                compare = comparator.compare( get( key ), otherMap.get( key ) );
                if ( compare != 0 )
                {
                    return compare;
                }
            }
        }
        return compare;
    }

    @Override
    public Comparison unsafeTernaryCompareTo( VirtualValue other, TernaryComparator comparator )
    {
        MapValue otherMap = (MapValue) other;
        int size = size();
        int compare = Integer.compare( size, otherMap.size() );
        if ( compare == 0 )
        {
            String[] thisKeys = StreamSupport.stream( keySet().spliterator(), false).toArray( String[]::new  );
            Arrays.sort( thisKeys, String::compareTo );
            String[] thatKeys = StreamSupport.stream( otherMap.keySet().spliterator(), false).toArray( String[]::new  );
            Arrays.sort( thatKeys, String::compareTo );
            for ( int i = 0; i < size; i++ )
            {
                compare = thisKeys[i].compareTo( thatKeys[i] );
                if ( compare != 0 )
                {
                    return Comparison.from( compare );
                }
            }

            for ( int i = 0; i < size; i++ )
            {
                String key = thisKeys[i];
                Comparison comparison = comparator.ternaryCompare( get( key ), otherMap.get( key ) );
                if ( comparison != Comparison.EQUAL )
                {
                    return comparison;
                }
            }
        }
        return Comparison.from( compare );
    }

    @Override
    public Equality ternaryEquals( AnyValue other )
    {
        assert other != null : "null values are not supported, use NoValue.NO_VALUE instead";
        if ( other == NO_VALUE )
        {
            return Equality.UNDEFINED;
        }
        else if ( !(other instanceof MapValue) )
        {
            return Equality.FALSE;
        }
        MapValue otherMap = (MapValue) other;
        int size = size();
        if ( size != otherMap.size() )
        {
            return Equality.FALSE;
        }
        String[] thisKeys = StreamSupport.stream( keySet().spliterator(), false ).toArray( String[]::new );
        Arrays.sort( thisKeys, String::compareTo );
        String[] thatKeys = StreamSupport.stream( otherMap.keySet().spliterator(), false ).toArray( String[]::new );
        Arrays.sort( thatKeys, String::compareTo );
        for ( int i = 0; i < size; i++ )
        {
            if ( thisKeys[i].compareTo( thatKeys[i] ) != 0 )
            {
                return Equality.FALSE;
            }
        }
        Equality equalityResult = Equality.TRUE;
        for ( int i = 0; i < size; i++ )
        {
            String key = thisKeys[i];
            AnyValue thisValue = get( key );
            AnyValue otherValue = otherMap.get( key );

            Equality equality = thisValue.ternaryEquals( otherValue );
            if ( equality == Equality.UNDEFINED )
            {
                equalityResult = Equality.UNDEFINED;
            }
            else if ( equality == Equality.FALSE )
            {
                return Equality.FALSE;
            }
        }
        return equalityResult;
    }

    @Override
    public  T map( ValueMapper mapper )
    {
        return mapper.mapMap( this );
    }

    public abstract  void foreach( ThrowingBiConsumer f ) throws E;

    public abstract boolean containsKey( String key );

    public abstract AnyValue get( String key );

    public MapValue filter( BiFunction filterFunction )
    {
        return new FilteringMapValue( this, filterFunction );
    }

    public MapValue updatedWith( String key, AnyValue value )
    {
        AnyValue current = get( key );
        if ( current.equals( value ) )
        {
            return this;
        }
        else if ( current == NO_VALUE )
        {
            return new UpdatedMapValue( this, new String[]{key}, new AnyValue[]{value} );
        }
        else
        {
            return new MappedMapValue( this, ( k, v ) -> {
                if ( k.equals( key ) )
                {
                    return value;
                }
                else
                {
                    return v;
                }
            } );
        }
    }

    public MapValue updatedWith(  MapValue other )
    {
        return new CombinedMapValue( this, other );
    }

    @Override
    public String getTypeName()
    {
        return "Map";
    }

    @Override
    public String toString()
    {
        StringBuilder sb = new StringBuilder( getTypeName() + "{" );
        final String[] sep = new String[]{""};
        foreach( ( key, value ) ->
        {
            sb.append( sep[0] );
            sb.append( key );
            sb.append( " -> " );
            sb.append( value );
            sep[0] = ", ";
        } );
        sb.append( '}' );
        return sb.toString();
    }

    @Override
    protected long estimatedPayloadSize()
    {
        //rough estimate
        return size() * 150;
    }

    public abstract int size();
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy