Authorize.scala

124 lines | 5.069 kB Blame History Raw Download
package org.keycloak.gatling

import java.text.SimpleDateFormat
import java.util.{Collections, Date}

import akka.actor.ActorDSL.actor
import akka.actor.ActorRef
import io.gatling.core.action.Interruptable
import io.gatling.core.action.builder.ActionBuilder
import io.gatling.core.config.Protocols
import io.gatling.core.result.writer.DataWriterClient
import io.gatling.core.session._
import io.gatling.core.validation._
import org.jboss.logging.Logger
import org.keycloak.adapters.KeycloakDeploymentBuilder
import org.keycloak.adapters.spi.AuthOutcome
import org.keycloak.adapters.spi.HttpFacade.Cookie
import org.keycloak.common.enums.SslRequired
import org.keycloak.representations.adapters.config.AdapterConfig

import scala.collection.JavaConverters._

case class AuthorizeAttributes(
  requestName: Expression[String],
  uri: Expression[String],
  cookies: Expression[List[Cookie]],
  sslRequired: SslRequired = SslRequired.EXTERNAL,
  resource: Option[Expression[String]] = None,
  secret: Option[Expression[String]] = None,
  isPublic: Option[Expression[Boolean]] = None,
  realm: Option[Expression[String]] = None,
  realmKey: Option[String] = None,
  authServerUrl: Expression[String] = _ => Failure("no server url")
) {
  def toAdapterConfig(session: Session) = {
    val adapterConfig = new AdapterConfig
    adapterConfig.setSslRequired(sslRequired.toString)

    adapterConfig.setResource( resource match {
      case Some(expr) => expr(session).get
      case None => null
    })
    adapterConfig.setPublicClient( isPublic match {
      case Some(expr) => expr(session).get
      case None => false
    })
    adapterConfig.setCredentials( secret match {
      case Some(expr) => Collections.singletonMap("secret", expr(session).get)
      case None => null
    })
    adapterConfig.setRealm(realm match {
      case Some(expr) => expr(session).get
      case None => null
    })
    adapterConfig.setRealmKey(realmKey match {
      case Some(key) => key
      case None => null
    })
    adapterConfig.setAuthServerUrl(authServerUrl(session).get)

    adapterConfig
  }
}

class AuthorizeActionBuilder(attributes: AuthorizeAttributes) extends ActionBuilder {
  def newInstance(attributes: AuthorizeAttributes) = new AuthorizeActionBuilder(attributes)

  def sslRequired(sslRequired: SslRequired) = newInstance(attributes.copy(sslRequired = sslRequired))
  def resource(resource: Expression[String]) = newInstance(attributes.copy(resource = Option(resource)))
  def clientCredentials(secret: Expression[String]) = newInstance(attributes.copy(secret = Option(secret)))
  def publicClient(isPublic: Expression[Boolean]) = newInstance(attributes.copy(isPublic = Option(isPublic)))
  def realm(realm: Expression[String]) = newInstance(attributes.copy(realm = Option(realm)))
  def realmKey(realmKey: String) = newInstance(attributes.copy(realmKey = Option(realmKey)))
  def authServerUrl(authServerUrl: Expression[String]) = newInstance(attributes.copy(authServerUrl = authServerUrl))

  override def build(next: ActorRef, protocols: Protocols): ActorRef = {
    actor(actorName("authorize"))(new AuthorizeAction(attributes, next))
  }
}

object AuthorizeAction {
  val logger = Logger.getLogger(classOf[AuthorizeAction])

  def init(session: Session) : Session = {
    session.remove(MockRequestAuthenticator.KEY)
  }
}

class AuthorizeAction(
                       attributes: AuthorizeAttributes,
                       val next: ActorRef
                     ) extends Interruptable with ExitOnFailure with DataWriterClient {
  override def executeOrFail(session: Session): Validation[_] = {
    val facade = new MockHttpFacade()
    val deployment = KeycloakDeploymentBuilder.build(attributes.toAdapterConfig(session))
    val url = attributes.uri(session).get
    facade.request.setURI(if (attributes.isPublic.isDefined && attributes.isPublic.get(session).get) rewriteFragmentToQuery(url) else url)
    facade.request.setCookies(attributes.cookies(session).get.map(c => (c.getName, c)).toMap.asJava)
    var nextSession = session
    val requestAuth: MockRequestAuthenticator = session(MockRequestAuthenticator.KEY).asOption[MockRequestAuthenticator] match {
      case Some(ra) => ra
      case None =>
        val tmp = new MockRequestAuthenticator(facade, deployment, new MockTokenStore, -1, session.userId)
        nextSession = session.set(MockRequestAuthenticator.KEY, tmp)
        tmp
    }

    Blocking(() => {
      AuthorizeAction.logger.debugf("%s: Authenticating %s%n", new SimpleDateFormat("HH:mm:ss,SSS").format(new Date()).asInstanceOf[Any], session("username").as[Any], Unit)
      Stopwatch(() => requestAuth.authenticate())
        .check(result => result == AuthOutcome.AUTHENTICATED, result => {
          AuthorizeAction.logger.warnf("%s: Failed auth %s%n", new SimpleDateFormat("HH:mm:ss,SSS").format(new Date()).asInstanceOf[Any], session("username").as[Any], Unit)
          "AuthorizeAction: authenticate() failed with status: " + result.toString
        })
        .recordAndContinue(this, nextSession, attributes.requestName(session).get)
    })
  }

  def rewriteFragmentToQuery(str: String): String = {
    str.replaceFirst("#", "?")
  }
}