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

org.partiql.lang.eval.StructExprValue.kt Maven / Gradle / Ivy

There is a newer version: 1.0.0-perf.1
Show newest version
/*
 * Copyright 2019 Amazon.com, Inc. or its affiliates.  All rights reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License").
 *  You may not use this file except in compliance with the License.
 * A copy of the License is located at:
 *
 *      http://aws.amazon.com/apache2.0/
 *
 *  or in the "license" file accompanying this file. This file is distributed on an "AS IS" BASIS,
 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific
 *  language governing permissions and limitations under the License.
 */

package org.partiql.lang.eval

import com.amazon.ion.IonSystem
import org.partiql.lang.errors.ErrorCode

/** Indicates if a struct is ordered or not. */
enum class StructOrdering {
    UNORDERED,
    ORDERED
}

/**
 * Provides a [ExprValueType.STRUCT] implementation lazily backed by a sequence.
 */
internal open class StructExprValue(
    private val ion: IonSystem,
    private val ordering: StructOrdering,
    private val sequence: Sequence
) : BaseExprValue() {

    override val type = ExprValueType.STRUCT

    /** The backing data structured for operations that require materialization. */
    private data class Materialized(
        val bindings: Bindings,
        val ordinalBindings: OrdinalBindings,
        val orderedBindNames: OrderedBindNames?
    )

    private val materialized by lazy {
        val bindMap = HashMap()
        val bindList = ArrayList()
        val bindNames = ArrayList()
        sequence.forEach {
            val name = it.name?.stringValue() ?: errNoContext("Expected non-null name for lazy struct", errorCode = ErrorCode.EVALUATOR_UNEXPECTED_VALUE, internal = false)
            bindMap.putIfAbsent(name, it)
            if (ordering == StructOrdering.ORDERED) {
                bindList.add(it)
                bindNames.add(name)
            }
        }

        val bindings = Bindings.ofMap(bindMap)
        val ordinalBindings = OrdinalBindings.ofList(bindList)
        val orderedBindNames = when (ordering) {
            StructOrdering.ORDERED -> object : OrderedBindNames {
                override val orderedNames = bindNames
            }
            StructOrdering.UNORDERED -> null
        }

        Materialized(bindings, ordinalBindings, orderedBindNames)
    }

    override val bindings: Bindings
        get() = materialized.bindings

    override val ordinalBindings: OrdinalBindings
        get() = materialized.ordinalBindings

    @Suppress("UNCHECKED_CAST")
    override fun  provideFacet(type: Class?): T? = when (type) {
        OrderedBindNames::class.java -> when (ordering) {
            StructOrdering.ORDERED -> materialized.orderedBindNames
            else -> null
        } as T?
        else -> null
    }

    override fun iterator() = sequence.iterator()

    override val ionValue by lazy {
        toIonValue(ion)
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy