
io.gatling.http.response.ResponseBuilder.scala Maven / Gradle / Ivy
The newest version!
/**
* Copyright 2011-2014 eBusiness Information, Groupe Excilys (www.ebusinessinformation.fr)
*
* 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 io.gatling.http.response
import java.nio.charset.Charset
import java.security.MessageDigest
import scala.collection.mutable.ArrayBuffer
import scala.math.max
import org.jboss.netty.buffer.ChannelBuffer
import com.ning.http.client.{ FluentCaseInsensitiveStringsMap, HttpResponseBodyPart, HttpResponseHeaders, HttpResponseStatus, Request }
import com.ning.http.client.providers.netty.ResponseBodyPart
import io.gatling.core.config.GatlingConfiguration.configuration
import io.gatling.core.util.StringHelper.bytes2Hex
import io.gatling.core.util.TimeHelper.nowMillis
import io.gatling.http.HeaderNames
import io.gatling.http.check.HttpCheck
import io.gatling.http.check.checksum.ChecksumCheck
import io.gatling.http.config.HttpProtocol
import io.gatling.http.util.HttpHelper.{ isCss, isHtml }
object ResponseBuilder {
val emptyBytes = Array.empty[Byte]
val emptyHeaders = new FluentCaseInsensitiveStringsMap
def newResponseBuilderFactory(checks: List[HttpCheck], responseTransformer: Option[ResponseTransformer], protocol: HttpProtocol): ResponseBuilderFactory = {
val checksumChecks = checks.collect {
case checksumCheck: ChecksumCheck => checksumCheck
}
val responseBodyUsageStrategies = checks.flatMap(_.responseBodyUsageStrategy).toSet
val storeBodyParts = !protocol.responsePart.discardResponseChunks || !responseBodyUsageStrategies.isEmpty
request: Request => new ResponseBuilder(request, checksumChecks, responseBodyUsageStrategies, responseTransformer, storeBodyParts, protocol.responsePart.fetchHtmlResources)
}
}
class ResponseBuilder(request: Request, checksumChecks: List[ChecksumCheck], bodyUsageStrategies: Set[ResponseBodyUsageStrategy], responseProcessor: Option[ResponseTransformer], storeBodyParts: Boolean, fetchHtmlResources: Boolean) {
val computeChecksums = !checksumChecks.isEmpty
var storeHtmlOrCss = false
var firstByteSent = nowMillis
var lastByteSent = 0L
var firstByteReceived = 0L
var lastByteReceived = 0L
private var status: Option[HttpResponseStatus] = None
private var headers: FluentCaseInsensitiveStringsMap = ResponseBuilder.emptyHeaders
private val chunks = new ArrayBuffer[ChannelBuffer]
private var digests: Map[String, MessageDigest] = initDigests()
def initDigests(): Map[String, MessageDigest] =
if (computeChecksums)
checksumChecks.foldLeft(Map.empty[String, MessageDigest]) { (map, check) =>
map + (check.algorithm -> MessageDigest.getInstance(check.algorithm))
}
else
Map.empty[String, MessageDigest]
def updateFirstByteSent() {
firstByteSent = nowMillis
}
def reset() {
firstByteSent = nowMillis
lastByteSent = 0L
firstByteReceived = 0L
lastByteReceived = 0L
status = None
headers = ResponseBuilder.emptyHeaders
chunks.clear()
digests = initDigests()
}
def updateLastByteSent() {
lastByteSent = nowMillis
}
def updateLastByteReceived() {
lastByteReceived = nowMillis
}
def accumulate(status: HttpResponseStatus) {
this.status = Some(status)
val now = nowMillis
firstByteReceived = now
lastByteReceived = now
}
def accumulate(headers: HttpResponseHeaders) {
this.headers = headers.getHeaders
storeHtmlOrCss = fetchHtmlResources && (isHtml(headers.getHeaders) || isCss(headers.getHeaders))
updateLastByteReceived()
}
def accumulate(bodyPart: HttpResponseBodyPart) {
updateLastByteReceived()
val channelBuffer = bodyPart.asInstanceOf[ResponseBodyPart].getChannelBuffer
if (storeBodyParts || storeHtmlOrCss)
chunks += channelBuffer
if (computeChecksums)
digests.values.foreach(_.update(channelBuffer.toByteBuffer))
}
def build: Response = {
// time measurement is imprecise due to multi-core nature
// moreover, ProgressListener might be called AFTER ChannelHandler methods
// ensure request doesn't end before starting
lastByteSent = max(lastByteSent, firstByteSent)
// ensure response doesn't start before request ends
firstByteReceived = max(firstByteReceived, lastByteSent)
// ensure response doesn't end before starting
lastByteReceived = max(lastByteReceived, firstByteReceived)
val checksums = digests.foldLeft(Map.empty[String, String]) { (map, entry) =>
val (algo, md) = entry
map + (algo -> bytes2Hex(md.digest))
}
val bodyLength = chunks.map(_.readableBytes).sum
val bodyUsages = bodyUsageStrategies.map(_.bodyUsage(bodyLength))
val charset = Option(headers.getFirstValue(HeaderNames.CONTENT_ENCODING)).map(Charset.forName).getOrElse(configuration.core.charset)
val body: ResponseBody =
if (bodyUsages.contains(ByteArrayResponseBodyUsage))
ByteArrayResponseBody(chunks, charset)
else if (bodyUsages.contains(InputStreamResponseBodyUsage) || bodyUsages.isEmpty)
InputStreamResponseBody(chunks, charset)
else
StringResponseBody(chunks, charset)
val rawResponse = HttpResponse(request, status, headers, body, checksums, bodyLength, charset, firstByteSent, lastByteSent, firstByteReceived, lastByteReceived)
responseProcessor match {
case Some(processor) => processor.applyOrElse(rawResponse, identity[Response])
case _ => rawResponse
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy