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

commonMain.expr.common.CompoundExpression.kt Maven / Gradle / Ivy

/*
 * Copyright (c) 2024, OpenSavvy and contributors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License 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 opensavvy.ktmongo.dsl.expr.common

import opensavvy.ktmongo.bson.BsonContext
import opensavvy.ktmongo.bson.BsonFieldWriter
import opensavvy.ktmongo.dsl.DangerousMongoApi
import opensavvy.ktmongo.dsl.KtMongoDsl
import opensavvy.ktmongo.dsl.LowLevelApi
import opensavvy.ktmongo.dsl.tree.CompoundNode
import opensavvy.ktmongo.dsl.utils.asImmutable

/**
 * A compound expression is an [Expression] that may have children.
 *
 * A compound expression may have `0..n` children.
 * Children are added by calling the [accept] function.
 *
 * This is also the supertype for all DSL scopes, since DSL scopes correspond to the ability to add children to
 * an expression.
 *
 * ### Implementation notes
 *
 * Prefer implementing [AbstractCompoundExpression] instead of implementing this interface directly.
 */
interface CompoundExpression : Expression, CompoundNode {

	/**
	 * Adds a new [expression] as a child of this one.
	 *
	 * Since [Expression] subtypes may generate arbitrary BSON, it is possible
	 * to use this method to inject arbitrary BSON (escaped or not) into any KtMongo DSL.
	 * Incorrect [Expression] implementations can create memory leaks,
	 * performance issues, data corruption or data leaks.
	 *
	 * We recommend against calling this function directly.
	 * Instead, you should find other functions declared on this object (possibly as extensions)
	 * that perform the operation you want in safe manner.
	 */
	@LowLevelApi
	@DangerousMongoApi
	@KtMongoDsl
	override fun accept(expression: Expression)

	companion object
}

/**
 * Abstract utility class to help implement [CompoundExpression].
 *
 * Learn more by reading [Expression], [AbstractExpression] and [CompoundExpression].
 */
abstract class AbstractCompoundExpression(
	context: BsonContext,
) : AbstractExpression(context), CompoundExpression {

	// region Sub-expression binding

	private val _children = ArrayList()

	@LowLevelApi
	protected val children: List
		get() = _children.asImmutable()

	@LowLevelApi
	@DangerousMongoApi
	@KtMongoDsl
	override fun accept(expression: Expression) {
		require(!frozen) { "This expression has already been frozen, it cannot accept the child expression $expression" }
		require(expression != this) { "Trying to add an expression to itself!" }

		val simplified = expression.simplify()

		if (simplified != null) {
			require(simplified !== this) { "Trying to add an expression to itself!" }
			simplified.freeze()
			_children += simplified
		}
	}

	// endregion
	// region Simplifications

	/**
	 * Simplifies a node based on its children.
	 *
	 * @param children The children of this expression, previously added with [accept].
	 * **They have already been simplified.**
	 * @see Expression.simplify
	 */
	@LowLevelApi
	protected open fun simplify(children: List): AbstractExpression? =
		this

	@LowLevelApi
	final override fun simplify(): AbstractExpression? =
		simplify(children)

	// endregion
	// region Writing

	/**
	 * Writes a node alongside its children.
	 *
	 * @param children The children of this expression, previously added with [accept].
	 * **They have already been simplified**.
	 * @see AbstractExpression.write
	 */
	@LowLevelApi
	protected open fun write(writer: BsonFieldWriter, children: List) {
		for (child in children) {
			check(this !== child) { "Trying to write myself as my own child!" }
			child.writeTo(writer)
		}
	}

	@LowLevelApi
	final override fun write(writer: BsonFieldWriter) {
		write(writer, children)
	}

	// endregion

	companion object
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy