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

org.neo4j.codegen.Expression Maven / Gradle / Ivy

/*
 * 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.codegen;

import java.lang.reflect.Modifier;
import java.util.Arrays;

import org.neo4j.values.AnyValue;

import static org.neo4j.codegen.TypeReference.BOOLEAN;
import static org.neo4j.codegen.TypeReference.DOUBLE;
import static org.neo4j.codegen.TypeReference.INT;
import static org.neo4j.codegen.TypeReference.LONG;
import static org.neo4j.codegen.TypeReference.OBJECT;
import static org.neo4j.codegen.TypeReference.VALUE;
import static org.neo4j.codegen.TypeReference.VOID;
import static org.neo4j.codegen.TypeReference.arrayOf;
import static org.neo4j.codegen.TypeReference.typeReference;

public abstract class Expression extends ExpressionTemplate
{
    public static final Expression TRUE = new Constant( BOOLEAN, Boolean.TRUE )
    {
        @Override
        Expression not()
        {
            return FALSE;
        }
    };
    public static final Expression FALSE = new Constant( BOOLEAN, Boolean.FALSE )
    {
        @Override
        Expression not()
        {
            return TRUE;
        }
    };
    public static final Expression NULL = new Constant( OBJECT, null );

    protected Expression( TypeReference type )
    {
        super( type );
    }

    public abstract void accept( ExpressionVisitor visitor );

    static final Expression SUPER = new Expression( OBJECT )
    {
        @Override
        public void accept( ExpressionVisitor visitor )
        {
            visitor.loadThis( "super" );
        }
    };

    public static final Expression EMPTY = new Expression( VOID )
    {
        @Override
        public void accept( ExpressionVisitor visitor )
        {
            //do nothing
        }
    };

    public static Expression gt( final Expression lhs, final Expression rhs )
    {
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.gt( lhs, rhs );
            }

            @Override
            Expression not()
            {
                return lte( lhs, rhs );
            }
        };
    }

    public static Expression gte( final Expression lhs, final Expression rhs )
    {
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.gte( lhs, rhs );
            }

            @Override
            Expression not()
            {
                return lt( lhs, rhs );
            }
        };
    }

    public static Expression lt( final Expression lhs, final Expression rhs )
    {
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.lt( lhs, rhs );
            }

            @Override
            Expression not()
            {
                return gte( lhs, rhs );
            }
        };
    }

    public static Expression lte( final Expression lhs, final Expression rhs )
    {
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.lte( lhs, rhs );
            }

            @Override
            Expression not()
            {
                return gt( lhs, rhs );
            }
        };
    }

    public static Expression and( final Expression lhs, final Expression rhs )
    {
        if ( lhs == FALSE || rhs == FALSE )
        {
            return FALSE;
        }
        if ( lhs == TRUE )
        {
            return rhs;
        }
        if ( rhs == TRUE )
        {
            return lhs;
        }
        Expression[] expressions;
        if ( lhs instanceof And )
        {
            if ( rhs instanceof And )
            {
                expressions = expressions( ((And) lhs).expressions, ((And) rhs).expressions );
            }
            else
            {
                expressions = expressions( ((And) lhs).expressions, rhs );
            }
        }
        else if ( rhs instanceof And )
        {
            expressions = expressions( lhs, ((And) rhs).expressions );
        }
        else
        {
            expressions = new Expression[] {lhs, rhs};
        }
        return new And( expressions );
    }

    public static Expression or( final Expression lhs, final Expression rhs )
    {
        if ( lhs == TRUE || rhs == TRUE )
        {
            return TRUE;
        }
        if ( lhs == FALSE )
        {
            return rhs;
        }
        if ( rhs == FALSE )
        {
            return lhs;
        }
        Expression[] expressions;
        if ( lhs instanceof Or )
        {
            if ( rhs instanceof Or )
            {
                expressions = expressions( ((Or) lhs).expressions, ((Or) rhs).expressions );
            }
            else
            {
                expressions = expressions( ((Or) lhs).expressions, rhs );
            }
        }
        else if ( rhs instanceof Or )
        {
            expressions = expressions( lhs, ((Or) rhs).expressions );
        }
        else
        {
            expressions = new Expression[] {lhs, rhs};
        }
        return new Or( expressions );
    }

    private static class And extends Expression
    {
        private final Expression[] expressions;

        And( Expression[] expressions )
        {
            super( BOOLEAN );
            this.expressions = expressions;
        }

        @Override
        public void accept( ExpressionVisitor visitor )
        {
            visitor.and( expressions );
        }
    }

    private static class Or extends Expression
    {
        private final Expression[] expressions;

        Or( Expression[] expressions )
        {
            super( BOOLEAN );
            this.expressions = expressions;
        }

        @Override
        public void accept( ExpressionVisitor visitor )
        {
            visitor.or( expressions );
        }
    }

    private static Expression[] expressions( Expression[] some, Expression[] more )
    {
        Expression[] result = Arrays.copyOf( some, some.length + more.length );
        System.arraycopy( more, 0, result, some.length, more.length );
        return result;
    }

    private static Expression[] expressions( Expression[] some, Expression last )
    {
        Expression[] result = Arrays.copyOf( some, some.length + 1 );
        result[some.length] = last;
        return result;
    }

    private static Expression[] expressions( Expression first, Expression[] more )
    {
        Expression[] result = new Expression[more.length + 1];
        result[0] = first;
        System.arraycopy( more, 0, result, 1, more.length );
        return result;
    }

    public static Expression equal( final Expression lhs, final Expression rhs )
    {
        if ( lhs == NULL )
        {
            if ( rhs == NULL )
            {
                return constant( Boolean.TRUE );
            }
            else
            {
                return isNull( rhs );
            }
        }
        else if ( rhs == NULL )
        {
            return isNull( lhs );
        }
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.equal( lhs, rhs );
            }

            @Override
            Expression not()
            {
                return notEqual( lhs, rhs );
            }
        };
    }

    public static Expression isNull( final Expression expression )
    {
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.isNull( expression );
            }

            @Override
            Expression not()
            {
                return notNull( expression );
            }
        };
    }

    public static Expression notNull( final Expression expression )
    {
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.notNull( expression );
            }

            @Override
            Expression not()
            {
                return isNull( expression );
            }
        };
    }

    public static Expression notEqual( final Expression lhs, final Expression rhs )
    {
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.notEqual( lhs, rhs );
            }

            @Override
            Expression not()
            {
                return equal( lhs, rhs );
            }
        };
    }

    public static Expression load( final LocalVariable variable )
    {
        return new Expression( variable.type() )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.load( variable );
            }
        };
    }

    public static Expression add( final Expression lhs, final Expression rhs )
    {
        if ( !lhs.type.equals( rhs.type ) )
        {
            throw new IllegalArgumentException(
                    String.format( "Cannot add variables with different types. LHS %s, RHS %s", lhs.type.simpleName(),
                            rhs.type.simpleName() ) );
        }

        return new Expression( lhs.type )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.add( lhs, rhs );
            }
        };
    }

    public static Expression subtract( final Expression lhs, final Expression rhs )
    {
        if ( !lhs.type.equals( rhs.type ) )
        {
            throw new IllegalArgumentException(
                    String.format(
                            "Cannot subtract variables with different types. LHS %s, RHS %s",
                            lhs.type.simpleName(),
                            rhs.type.simpleName() ) );
        }
        return new Expression( lhs.type )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.subtract( lhs, rhs );
            }
        };
    }

    public static Expression multiply( final Expression lhs, final Expression rhs )
    {
        if ( !lhs.type.equals( rhs.type ) )
        {
            throw new IllegalArgumentException(
                    String.format(
                            "Cannot multiply variables with different types. LHS %s, RHS %s",
                            lhs.type.simpleName(),
                            rhs.type.simpleName() ) );
        }
        return new Expression( lhs.type )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.multiply( lhs, rhs );
            }
        };
    }

    public static Expression constantInt( int value )
    {
        return constant(value);
    }

    public static Expression constantLong( long value )
    {
        return constant(value);
    }

    public static Expression constant( final Object value )
    {
        TypeReference reference;
        if ( value == null )
        {
            return NULL;
        }
        else if ( value instanceof String )
        {
            reference = TypeReference.typeReference( String.class );
        }
        else if ( value instanceof Long )
        {
            reference = LONG;
        }
        else if ( value instanceof Integer )
        {
            reference = INT;
        }
        else if ( value instanceof Double )
        {
            reference = DOUBLE;
        }
        else if ( value instanceof Boolean )
        {
            return (Boolean) value ? TRUE : FALSE;
        }
        else if ( value instanceof AnyValue )
        {
            reference = VALUE;
        }
        else
        {
            throw new IllegalArgumentException( "Not a valid constant: " + value );
        }

        return new Expression( reference )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.constant( value );
            }
        };
    }

    private static class Constant extends Expression
    {
        private final Object value;

        Constant( TypeReference type, Object value )
        {
            super( type );
            this.value = value;
        }

        @Override
        public void accept( ExpressionVisitor visitor )
        {
            visitor.constant( value );
        }
    }

    //TODO deduce type from constants
    public static Expression newArray( TypeReference baseType, Expression... constants )
    {
        return new Expression( arrayOf( baseType ) )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.newArray( baseType, constants );
            }
        };
    }

    /** get instance field */
    public static Expression get( final Expression target, final FieldReference field )
    {
        return new Expression( field.type() )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.getField( target, field );
            }
        };
    }

    /** box expression */
    public static Expression box( final Expression expression )
    {
        TypeReference type = expression.type;
        if ( type.isPrimitive() )
        {
            switch ( type.name() )
            {
            case "byte":
                type = TypeReference.typeReference( Byte.class );
                break;
            case "short":
                type = TypeReference.typeReference( Short.class );
                break;
            case "int":
                type = TypeReference.typeReference( Integer.class );
                break;
            case "long":
                type = TypeReference.typeReference( Long.class );
                break;
            case "char":
                type = TypeReference.typeReference( Character.class );
                break;
            case "boolean":
                type = TypeReference.typeReference( Boolean.class );
                break;
            case "float":
                type = TypeReference.typeReference( Float.class );
                break;
            case "double":
                type = TypeReference.typeReference( Double.class );
                break;
            default:
                break;
            }
        }
        return new Expression( type )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.box( expression );
            }
        };
    }

    /** unbox expression */
    public static Expression unbox( final Expression expression )
    {
        TypeReference type;
        switch ( expression.type.fullName() )
        {
        case "java.lang.Byte":
            type = TypeReference.typeReference( byte.class );
            break;
        case "java.lang.Short":
            type = TypeReference.typeReference( short.class );
            break;
        case "java.lang.Integer":
            type = TypeReference.typeReference( int.class );
            break;
        case "java.lang.Long":
            type = TypeReference.typeReference( long.class );
            break;
        case "java.lang.Character":
            type = TypeReference.typeReference( char.class );
            break;
        case "java.lang.Boolean":
            type = TypeReference.typeReference( boolean.class );
            break;
        case "java.lang.Float":
            type = TypeReference.typeReference( float.class );
            break;
        case "java.lang.Double":
            type = TypeReference.typeReference( double.class );
            break;
        default:
            throw new IllegalStateException( "Cannot unbox " + expression.type.fullName() );
        }
        return new Expression( type )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.unbox( expression );
            }
        };
    }

    /** get static field */
    public static Expression getStatic( final FieldReference field )
    {
        return new Expression( field.type() )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.getStatic( field );
            }
        };
    }

    public static Expression ternary( final Expression test, final Expression onTrue, final Expression onFalse )
    {
        TypeReference reference = onTrue.type.equals( onFalse.type ) ? onTrue.type : OBJECT;
        return new Expression( reference )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.ternary( test, onTrue, onFalse );
            }
        };
    }

    public static Expression invoke(
            final Expression target, final MethodReference method,
            final Expression... arguments )
    {
        return new Expression( method.returns() )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.invoke( target, method, arguments );
            }
        };
    }

    public static Expression invoke( final MethodReference method, final Expression... parameters )
    {
        return new Expression( method.returns() )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.invoke( method, parameters );
            }
        };
    }

    public static Expression invokeSuper( TypeReference parent, final Expression... parameters )
    {
        TypeReference[] parameterTypes = new TypeReference[parameters.length];
        for ( int i = 0; i < parameters.length; i++ )
        {
            parameterTypes[i] = parameters[i].type();
        }

        return new Expression( OBJECT )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.invoke( Expression.SUPER,
                        new MethodReference( parent, "", VOID, Modifier.PUBLIC, parameterTypes ), parameters);
            }
        };
    }

    public static Expression cast( Class type, Expression expression )
    {
        return cast( typeReference( type ), expression );
    }

    public static Expression instanceOf( final TypeReference typeToCheck, Expression expression )
    {
        return new Expression( typeReference( boolean.class ) )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.instanceOf( typeToCheck, expression );
            }
        };
    }

    public static Expression cast( final TypeReference type, Expression expression )
    {
        return new Expression( type )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.cast( type, expression );
            }
        };
    }

    public static Expression newInstance( Class type )
    {
        return newInstance( typeReference( type ) );
    }

    public static Expression newInstance( final TypeReference type )
    {
        return new Expression( type )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.newInstance( type );
            }
        };
    }

    public static Expression not( final Expression expression )
    {
        return expression.not();
    }

    Expression not()
    {
        return notExpr( this );
    }

    private static Expression notExpr( final Expression expression )
    {
        assert expression.type == BOOLEAN : "Can only apply not() to boolean expressions";
        return new Expression( BOOLEAN )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.not( expression );
            }

            @Override
            Expression not()
            {
                return expression;
            }
        };
    }

    public static Expression toDouble( final Expression expression )
    {
        return new Expression( DOUBLE )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.longToDouble( expression );
            }
        };
    }

    public static Expression pop( Expression expression )
    {
        return new Expression( expression.type )
        {
            @Override
            public void accept( ExpressionVisitor visitor )
            {
                visitor.pop( expression );
            }
        };
    }

    @Override
    Expression materialize( CodeBlock method )
    {
        return this;
    }

    @Override
    void templateAccept( CodeBlock method, ExpressionVisitor visitor )
    {
        throw new UnsupportedOperationException( "simple expressions should not be invoked as templates" );
    }

    @Override
    public String toString()
    {
        StringBuilder result = new StringBuilder().append( "Expression[" );
        accept( new ExpressionToString( result ) );
        return result.append( ']' ).toString();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy