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

commonMain.com.algolia.search.model.rule.Consequence.kt Maven / Gradle / Ivy

package com.algolia.search.model.rule

import com.algolia.search.model.ObjectID
import com.algolia.search.model.response.ResponseSearch
import com.algolia.search.model.search.Query
import com.algolia.search.serialize.KSerializerObjectIDs
import com.algolia.search.serialize.internal.Json
import com.algolia.search.serialize.internal.JsonNoDefaults
import com.algolia.search.serialize.internal.Key
import com.algolia.search.serialize.internal.asJsonInput
import com.algolia.search.serialize.internal.asJsonOutput
import com.algolia.search.serialize.internal.jsonArrayOrNull
import com.algolia.search.serialize.internal.jsonObjectOrNull
import com.algolia.search.serialize.internal.jsonPrimitiveOrNull
import com.algolia.search.serialize.internal.toJsonNoDefaults
import kotlinx.serialization.ExperimentalSerializationApi
import kotlinx.serialization.KSerializer
import kotlinx.serialization.SerialName
import kotlinx.serialization.Serializable
import kotlinx.serialization.Serializer
import kotlinx.serialization.builtins.ListSerializer
import kotlinx.serialization.encoding.Decoder
import kotlinx.serialization.encoding.Encoder
import kotlinx.serialization.json.JsonElement
import kotlinx.serialization.json.JsonObject
import kotlinx.serialization.json.booleanOrNull
import kotlinx.serialization.json.buildJsonObject
import kotlinx.serialization.json.decodeFromJsonElement
import kotlinx.serialization.json.encodeToJsonElement
import kotlinx.serialization.json.jsonObject
import kotlinx.serialization.json.jsonPrimitive
import kotlinx.serialization.json.put

@Serializable(Consequence.Companion::class)
public data class Consequence(
    /**
     * Names of facets to which automatic filtering must be applied; they must match the facet name of a facet value
     * placeholder in the query pattern.
     * Ex. facetName1, facetName2. You can specify a score: facetName1, facetName2.
     */
    @SerialName(Key.AutomaticFacetFilters) val automaticFacetFilters: List? = null,
    /**
     * Same as [automaticFacetFilters], but the engine treats the filters as optional.
     * Behaves like [Query.optionalFilters].
     */
    val automaticOptionalFacetFilters: List? = null,
    /**
     * Describes incremental edits to be made to the query string. You can't do both this and [edits] at the same time.
     */
    val edits: List? = null,
    /**
     * When providing a [Query.query], it replaces the entire query string. You can't do both this and [edits] at the
     * same time.
     */
    val query: Query? = null,
    /**
     * Objects to promote as hits.
     */
    val promote: List? = null,
    /**
     * Whether promoted results should match the filters of the current search.
     */
    val filterPromotes: Boolean? = null,
    /**
     * Objects to hide from hits.
     */
    val hide: List? = null,
    /**
     * Custom JSON object that will be appended to the [ResponseSearch.userData].
     * This object is not interpreted by the API. It is limited to 1kB of minified JSON.
     */
    val userData: JsonObject? = null,
    /**
     * Content defining how the search interface should be rendered.
     * A default value for this can be set via settings.
     */
    val renderingContent: RenderingContent? = null,
) {

    @OptIn(ExperimentalSerializationApi::class)
    @Serializer(Consequence::class)
    public companion object : KSerializer {

        private val serializer = ListSerializer(AutomaticFacetFilters.serializer())

        private fun MutableMap.putFilters(key: String, filters: List?) {
            filters?.let { put(key, JsonNoDefaults.encodeToJsonElement(serializer, it)) }
        }

        private fun JsonObject.getFilters(key: String): List? {
            return this[key]?.jsonArrayOrNull?.let { JsonNoDefaults.decodeFromJsonElement(serializer, it) }
        }

        private fun JsonObject.getPromotions(): List? {
            return this[Key.Promote]?.jsonArrayOrNull?.let {
                Json.decodeFromJsonElement(ListSerializer(Promotion.serializer()), it)
            }
        }

        private fun JsonObject.getObjectIDs(): List? {
            return this[Key.Hide]?.jsonArrayOrNull?.let { Json.decodeFromJsonElement(KSerializerObjectIDs, it) }
        }

        private fun JsonObject.getEdits(): List? {
            return this[Key.Query]?.jsonObjectOrNull?.let { local ->
                local[Key.Edits]?.jsonArrayOrNull?.let { Json.decodeFromJsonElement(ListSerializer(Edit), it) }
                    ?: local[Key.RemoveLowercase]?.jsonArrayOrNull?.map { Edit(it.jsonPrimitive.content) }
            }
        }

        private fun JsonObject.extractQuery(edits: List?): Query? {
            val modified = toMutableMap().apply {
                if (edits != null) remove(Key.Query)
                remove(Key.AutomaticFacetFilters)
                remove(Key.AutomaticOptionalFacetFilters)
                remove(Key.RenderingContent)
            }
            return if (modified.isNotEmpty()) {
                JsonNoDefaults.decodeFromJsonElement(Query.serializer(), JsonObject(modified))
            } else null
        }

        private fun JsonObject.getRenderingContent(): RenderingContent? {
            return this[Key.RenderingContent]?.jsonObjectOrNull?.let { Json.decodeFromJsonElement(it) }
        }

        override fun serialize(encoder: Encoder, value: Consequence) {
            val params = mutableMapOf().apply {
                putFilters(Key.AutomaticFacetFilters, value.automaticFacetFilters)
                putFilters(Key.AutomaticOptionalFacetFilters, value.automaticOptionalFacetFilters)
                value.query?.toJsonNoDefaults()?.let { putAll(it) }
                if (value.edits != null) put(
                    Key.Query,
                    buildJsonObject {
                        put(Key.Edits, JsonNoDefaults.encodeToJsonElement(ListSerializer(Edit), value.edits))
                    }
                )
            }
            val json = buildJsonObject {
                if (params.isNotEmpty()) put(Key.Params, JsonObject(params))
                value.promote?.let {
                    put(Key.Promote, JsonNoDefaults.encodeToJsonElement(ListSerializer(Promotion.serializer()), it))
                }
                value.hide?.let {
                    put(Key.Hide, JsonNoDefaults.encodeToJsonElement(KSerializerObjectIDs, it))
                }
                value.userData?.let { put(Key.UserData, it) }
                value.filterPromotes?.let { put(Key.FilterPromotes, it) }
                value.renderingContent?.let {
                    put(Key.RenderingContent, JsonNoDefaults.encodeToJsonElement(it))
                }
            }

            encoder.asJsonOutput().encodeJsonElement(json)
        }

        override fun deserialize(decoder: Decoder): Consequence {
            val json = decoder.asJsonInput().jsonObject
            val params = json[Key.Params]?.jsonObjectOrNull
            val automaticFacetFilters = params?.getFilters(Key.AutomaticFacetFilters)
            val automaticOptionalFacetFilters = params?.getFilters(Key.AutomaticOptionalFacetFilters)
            val promote = json.getPromotions()
            val hide = json.getObjectIDs()
            val userData = json[Key.UserData]?.jsonObjectOrNull
            val edits = params?.getEdits()
            val query = params?.extractQuery(edits)
            val filterPromotes = json[Key.FilterPromotes]?.jsonPrimitiveOrNull?.booleanOrNull
            val renderingContent = json.getRenderingContent()

            return Consequence(
                automaticFacetFilters = automaticFacetFilters,
                automaticOptionalFacetFilters = automaticOptionalFacetFilters,
                query = query,
                promote = promote,
                hide = hide,
                userData = userData,
                edits = edits,
                filterPromotes = filterPromotes,
                renderingContent = renderingContent,
            )
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy