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

io.github.graphglue.model.Node.kt Maven / Gradle / Ivy

Go to download

A library to develop annotation-based code-first GraphQL servers using GraphQL Kotlin, Spring Boot and Neo4j - excluding Spring GraphQL server dependencies

The newest version!
package io.github.graphglue.model

import com.expediagroup.graphql.generator.annotations.GraphQLDescription
import com.expediagroup.graphql.generator.annotations.GraphQLIgnore
import com.expediagroup.graphql.generator.annotations.GraphQLName
import com.expediagroup.graphql.generator.scalars.ID
import com.fasterxml.jackson.databind.ObjectMapper
import graphql.schema.DataFetchingEnvironment
import io.github.graphglue.data.LazyLoadingContext
import io.github.graphglue.data.execution.FieldFetchingContext
import io.github.graphglue.data.execution.NodeQueryEntry
import io.github.graphglue.data.execution.NodeQueryParser
import io.github.graphglue.data.execution.PartialNodeQuery
import io.github.graphglue.definition.NodeDefinitionCollection
import io.github.graphglue.definition.RelationshipFieldDefinition
import io.github.graphglue.graphql.schema.FieldDataFetchingEnvironment
import io.github.graphglue.model.property.LazyLoadingSubqueryGenerator
import io.github.graphglue.model.property.NodePropertyDelegate
import io.github.graphglue.model.property.NodeSetPropertyDelegate
import io.github.graphglue.model.property.PropertyDelegate
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.data.annotation.Transient
import org.springframework.data.neo4j.core.schema.GeneratedValue
import org.springframework.data.neo4j.core.schema.Id
import kotlin.properties.PropertyDelegateProvider
import kotlin.reflect.KProperty
import kotlin.reflect.KProperty1

/**
 * Name of the bean used as id generator for [Node]s
 */
const val NODE_ID_GENERATOR_BEAN = "nodeIdGenerator"

/**
 * Base class for all Nodes
 * This is always added to the schema
 * All domain entities which can be retrieved via the api
 * and should be persisted in the database should inherit from this class
 * Two nodes are equal iff they have the same id, or, if no id is present yet on both, if they are the same object.
 */
@DomainNode
@AdditionalFilter("idIdFilter")
@GraphQLDescription("Base class of all nodes")
abstract class Node {

    /**
     * Id of this node, `null` if not persisted in the database yet
     */
    @Id
    @GeneratedValue(generatorRef = NODE_ID_GENERATOR_BEAN)
    internal var id: String? = null

    /**
     * Flag to indicate if this node has been saved in the database
     * If true, this node must not be saved again
     */
    @Transient
    internal var isSaved: Boolean = false

    /**
     * Readonly wrapper for the id
     * If `null`, the node has not been persisted in the database yet
     */
    @GraphQLIgnore
    val rawId
        get() = if (isPersisted) id else null

    /**
     * The id of the node as seen in the GraphQL API
     * @throws Exception if this node has not been persisted yet and therefore has no id
     */
    @GraphQLName("id")
    @GraphQLDescription("The unique id of this node")
    val graphQLId: ID
        get() = ID(id!!)

    /**
     * Context necessary for lazy-loading
     */
    @Transient
    internal var lazyLoadingContext: LazyLoadingContext? = null

    /**
     * True if this node is persisted in the database
     */
    @GraphQLIgnore
    val isPersisted: Boolean
        get() = id != null && lazyLoadingContext != null

    /**
     * Lookup for all node properties
     * Trades memory (additional HashMap) for a cleaner and more extensible way to lookup the delegates
     * (compared to reflection)
     * Name of property as key
     */
    @Transient
    internal val propertyLookup: MutableMap> = mutableMapOf()

    /**
     * Cached fetched values for fields
     * Key is the path to the resultKey
     * If entry is not found, it has not been fetched yet
     */
    @Transient
    internal var fieldCache: MutableMap = mutableMapOf()

    /**
     * Order fields for sorting and cursor generation fetched from the database
     */
    @Transient
    internal var orderFields: MutableMap? = null

    /**
     * Creates a new node property used for many sides
     *
     * @param T value type
     * @return a provider for the property delegate
     */
    protected fun  NodeSetProperty(): PropertyDelegateProvider> {
        return NodeSetPropertyProvider()
    }

    /**
     * Creates a new node property used for one sides
     *
     * @param T value type
     * @return a provider for the property delegate
     */
    protected fun  NodeProperty(): PropertyDelegateProvider> {
        return NodePropertyProvider()
    }

    /**
     * Gets the result of a GraphQL query
     * Uses the cache to obtain the result, and if no cache entry was found, creates a
     * new database query and executes it
     *
     * @param nodeQueryParser used to obtain the [NodeDefinitionCollection] and [ObjectMapper]
     * @param dataFetchingEnvironment environment to fetch data, used to parse subtree of fetched nodes
     * @return the result, including a new local context
     */
    internal suspend fun getFromGraphQL(
        @Autowired
        @GraphQLIgnore
        nodeQueryParser: NodeQueryParser,
        dataFetchingEnvironment: DataFetchingEnvironment,
    ): Any? {
        val fieldDefinition = (dataFetchingEnvironment as FieldDataFetchingEnvironment).fieldDefinition
        val cacheKey = dataFetchingEnvironment.executionStepInfo.path.keysOnly.joinToString("/")
        val lazyLoadingContext = requireNotNull(this.lazyLoadingContext) {
            "Cannot lazy-load field on not-loaded node"
        }
        if (cacheKey !in fieldCache) {
            val nodeDefinition = lazyLoadingContext.nodeDefinitionCollection.getNodeDefinition(this::class)
            val query = PartialNodeQuery(
                nodeDefinition, listOf(
                    fieldDefinition.createQueryEntry(
                        dataFetchingEnvironment,
                        FieldFetchingContext.from(dataFetchingEnvironment),
                        nodeQueryParser,
                        null
                    )
                )
            )
            lazyLoadingContext.nodeQueryEngine.execute(query, listOf(this))
        }
        if (cacheKey !in fieldCache) {
            error("Field $cacheKey not loaded")
        }
        return fieldDefinition.createGraphQLResult(fieldCache[cacheKey], lazyLoadingContext)
    }

    /**
     * Registers the result of a query
     * Used to cache the result of a query
     *
     * @param entry the entry of the query
     * @param queryResult the result of the query
     */
    internal fun registerQueryResult(entry: NodeQueryEntry<*>, queryResult: Any?) {
        fieldCache[entry.resultKeyPath] = queryResult
        if (entry.fieldDefinition.property != null) {
            val property = entry.fieldDefinition.property
            getProperty>(property).registerQueryResult(
                entry.fieldDefinition, queryResult
            )
        }
    }

    /**
     * Loads all nodes of a relationship
     * If the `loader` is provided, specified nested nodes are loaded too
     *
     * @param property defines the relation to load the nodes of
     * @param loader if provided used to define nested nodes to load
     * @return the result of the query and the query itself
     */
    internal suspend fun  loadNodesOfRelationship(
        property: KProperty1<*, *>, loader: (LazyLoadingSubqueryGenerator.() -> Unit)?
    ) {
        val lazyLoadingContext = lazyLoadingContext
        if (lazyLoadingContext != null) {
            val nodeDefinitionCollection = lazyLoadingContext.nodeDefinitionCollection
            val parentNodeDefinition = nodeDefinitionCollection.getNodeDefinition(this::class)
            val fieldDefinition =
                parentNodeDefinition.getFieldDefinitionOfProperty(property) as RelationshipFieldDefinition<*>
            val generator =
                LazyLoadingSubqueryGenerator(fieldDefinition, parentNodeDefinition, nodeDefinitionCollection)
            loader?.invoke(generator)
            val query = PartialNodeQuery(parentNodeDefinition, listOf(generator.toSubQuery()))
            lazyLoadingContext.nodeQueryEngine.execute(query, listOf(this))
        }
    }

    /**
     * Gets a node property from the lookup
     * May be changed in future to support extensibility
     *
     * @param property the property to lookup
     * @return the found property
     */
    @Suppress("UNCHECKED_CAST")
    internal fun > getProperty(property: KProperty1<*, *>): T {
        return propertyLookup[property.name]!! as T
    }

    final override fun equals(other: Any?): Boolean {
        return if (other !is Node) {
            false
        } else if (this === other) {
            true
        } else {
            this.id != null && this.id == other.id
        }
    }

    final override fun hashCode(): Int {
        return if (rawId != null) {
            rawId.hashCode()
        } else {
            super.hashCode()
        }
    }

    override fun toString(): String {
        return "${this::class.simpleName}(id=$id)"
    }
}

/**
 * Provider for [NodePropertyDelegate]s
 */
private class NodePropertyProvider : PropertyDelegateProvider> {

    /**
     * Creates a new [NodePropertyDelegate] and registers it to the [Node.propertyLookup]
     *
     * @param thisRef the parent node
     * @param property the property to delegate
     * @return the generated property delegate
     */
    override operator fun provideDelegate(thisRef: Node, property: KProperty<*>): NodePropertyDelegate {
        val nodeProperty = NodePropertyDelegate(
            thisRef, property as KProperty1<*, *>
        )
        thisRef.propertyLookup[property.name] = nodeProperty
        return nodeProperty
    }
}

/**
 * Provider for [NodeSetPropertyDelegate]s
 */
private class NodeSetPropertyProvider : PropertyDelegateProvider> {

    /**
     * Creates a new [NodeSetPropertyDelegate] and registers it to the [Node.propertyLookup]
     *
     * @param thisRef the parent node
     * @param property the property to delegate
     * @return the generated property delegate
     */
    override operator fun provideDelegate(thisRef: Node, property: KProperty<*>): NodeSetPropertyDelegate {
        val nodeSetPropertyDelegate = NodeSetPropertyDelegate(
            thisRef, property as KProperty1<*, *>
        )
        thisRef.propertyLookup[property.name] = nodeSetPropertyDelegate
        return nodeSetPropertyDelegate
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy