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

jvmAndNixMain.io.ktor.server.sessions.Sessions.kt Maven / Gradle / Ivy

Go to download

Ktor is a framework for quickly creating web applications in Kotlin with minimal effort.

There is a newer version: 3.0.0-rc-1
Show newest version
/*
* Copyright 2014-2021 JetBrains s.r.o and contributors. Use of this source code is governed by the Apache 2.0 license.
*/

package io.ktor.server.sessions

import io.ktor.server.application.*
import io.ktor.server.request.*
import io.ktor.server.response.*
import io.ktor.util.*
import io.ktor.util.logging.*

internal val SessionProvidersKey = AttributeKey>>("SessionProvidersKey")

internal val LOGGER = KtorSimpleLogger("io.ktor.server.sessions.Sessions")

/**
 * A plugin that provides a mechanism to persist data between different HTTP requests.
 * Typical use cases include storing a logged-in user's ID, the contents of a shopping basket,
 * or keeping user preferences on the client.
 * In Ktor, you can implement sessions by using cookies or custom headers,
 * choose whether to store session data on the server or pass it to the client,
 * sign and encrypt session data and more.
 *
 * You can learn more from [Sessions](https://ktor.io/docs/sessions.html).
 * @property providers list of session providers
 */
public val Sessions: RouteScopedPlugin = createRouteScopedPlugin("Sessions", ::SessionsConfig) {
    val providers = pluginConfig.providers.toList()

    application.attributes.put(SessionProvidersKey, providers)

    onCall { call ->
        // For each call, call each provider and retrieve session data if needed.
        // Capture data in the attribute's value
        val providerData = providers.associateBy({ it.name }) {
            it.receiveSessionData(call)
        }

        if (providerData.isEmpty()) {
            LOGGER.trace("No sessions found for ${call.request.uri}")
        } else {
            val sessions = providerData.keys.joinToString()
            LOGGER.trace("Sessions found for ${call.request.uri}: $sessions")
        }
        val sessionData = SessionData(providerData)
        call.attributes.put(SessionDataKey, sessionData)
    }

    // When response is being sent, call each provider to update/remove session data
    on(BeforeSend) { call ->

        /*
         If sessionData is not available it means response happened before Session plugin got a chance to deserialize the data.
         We should ignore this call in this case.

         An example would be CORS plugin responding with 403 Forbidden
         */
        val sessionData = call.attributes.getOrNull(SessionDataKey) ?: return@on

        sessionData.providerData.values.forEach { data ->
            LOGGER.trace("Sending session data for ${call.request.uri}: ${data.provider.name}")
            data.sendSessionData(call)
        }

        sessionData.commit()
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy