org.neo4j.values.virtual.MapValue Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of neo4j-values Show documentation
Show all versions of neo4j-values Show documentation
Neo4j property value system.
/*
* 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