com.yahoo.maha.api.jersey.MahaResource.scala Maven / Gradle / Ivy
// Copyright 2017, Yahoo Holdings Inc.
// Licensed under the terms of the Apache License 2.0. Please see LICENSE file in project root for terms.
package com.yahoo.maha.api.jersey
import java.util.UUID
import com.yahoo.maha.service.calcite.avatica.{AvaticaMahaJsonHandler, AvaticaMahaProtobufHandler, MahaAvaticaService}
import javax.servlet.http.HttpServletRequest
import javax.ws.rs.container.{AsyncResponse, ContainerRequestContext, Suspended}
import javax.ws.rs.core.{Context, MediaType}
import javax.ws.rs.{Path, Produces, _}
import com.yahoo.maha.core.bucketing.{BucketParams, UserInfo}
import com.yahoo.maha.core.request.{BaseRequest, ReportingRequest, RequestContext}
import com.yahoo.maha.core.{Schema, _}
import com.yahoo.maha.parrequest2.GeneralError
import com.yahoo.maha.service._
import com.yahoo.maha.service.output.{DebugRenderer, NoopDebugRenderer}
import com.yahoo.maha.service.utils.MahaConstants
import grizzled.slf4j.Logging
import org.apache.calcite.avatica.metrics.noop.NoopMetricsSystem
import org.apache.calcite.avatica.proto.Common.WireMessage
import org.apache.calcite.avatica.remote.{ProtobufHandler, ProtobufTranslationImpl}
import org.apache.commons.io.IOUtils
import org.apache.commons.lang3.StringUtils
import org.slf4j.MDC
import org.springframework.stereotype.Component
import scala.util.Try
@Path("/registry")
@Component
class MahaResource(mahaService: MahaService
, baseRequest: BaseRequest
, requestValidator: RequestValidator
, mahaRequestContextBuilder: MahaRequestContextBuilder
, debugRenderer: DebugRenderer = new NoopDebugRenderer
, debugUserListCSV: String = ""
, mahaAvaticaService: MahaAvaticaService
) extends Logging {
private[this] val debugUsers: Set[String] = debugUserListCSV.split(",").toSet
private[this] val defaultRequestCoordinator = DefaultRequestCoordinator(mahaService)
private[this] val mahaRequestProcessorFactory = MahaSyncRequestProcessorFactory(defaultRequestCoordinator
, mahaService
, mahaService.mahaRequestLogWriter
, mahaServiceMonitor = DefaultMahaServiceMonitor)
private[this] val defaultDebugRenderer = Option(debugRenderer)
@GET
@Path("/{registryName}/cubes")
@Produces(Array(MediaType.APPLICATION_JSON))
def getCubes(@PathParam("registryName") registryName: String): String = {
val domainjson: Option[String] = mahaService.getCubes(registryName)
if(domainjson.isDefined) {
domainjson.get
} else {
throw NotFoundException(Error(s"registry $registryName not found"))
}
}
@GET
@Path("/{registryName}/domain")
@Produces(Array(MediaType.APPLICATION_JSON))
def getDomain(@PathParam("registryName") registryName: String): String = {
val domainjson: Option[String] = mahaService.getDomain(registryName)
if(domainjson.isDefined) {
domainjson.get
} else {
throw NotFoundException(Error(s"registry $registryName not found"))
}
}
@GET
@Path("/{registryName}/domain/cubes/{cube}")
@Produces(Array(MediaType.APPLICATION_JSON))
def getDomainForCube(@PathParam("registryName") registryName: String, @PathParam("cube") cube: String): String = {
val domainjson: Option[String] = mahaService.getDomainForCube(registryName, cube)
if(domainjson.isDefined) {
domainjson.get
} else {
throw NotFoundException(Error(s"registry $registryName and cube $cube not found"))
}
}
@GET
@Path("/{registryName}/flattenDomain")
@Produces(Array(MediaType.APPLICATION_JSON))
def getFlattenDomain(@PathParam("registryName") registryName: String): String = {
val domainjson: Option[String] = mahaService.getFlattenDomain(registryName)
if(domainjson.isDefined) {
domainjson.get
} else {
throw NotFoundException(Error(s"registry $registryName not found"))
}
}
@GET
@Path("/{registryName}/flattenDomain/cubes/{cube}")
@Produces(Array(MediaType.APPLICATION_JSON))
def getFlattenDomainForCube(@PathParam("registryName") registryName: String, @PathParam("cube") cube: String): String = {
val domainjson: Option[String] = mahaService.getFlattenDomainForCube(registryName, cube)
if(domainjson.isDefined) {
domainjson.get
} else {
throw NotFoundException(Error(s"registry $registryName and cube $cube not found"))
}
}
@GET
@Path("/{registryName}/flattenDomain/cubes/{cube}/{revision}")
@Produces(Array(MediaType.APPLICATION_JSON))
def getFlattenDomainForCube(@PathParam("registryName") registryName: String, @PathParam("cube") cube: String, @PathParam("revision") revision: Int): String = {
val domainjson: Option[String] = mahaService.getFlattenDomainForCube(registryName, cube, Option(revision))
if(domainjson.isDefined) {
domainjson.get
} else {
throw NotFoundException(Error(s"registry $registryName and cube $cube with revision $revision not found"))
}
}
@GET
@Path("/{registryName}/fulldomain")
@Produces(Array(MediaType.APPLICATION_JSON))
def getRevisionedDomain(@PathParam("registryName") registryName: String): String = {
val domainjson: Option[String] = mahaService.getRevisionedDomain(registryName)
if(domainjson.isDefined) {
domainjson.get
} else {
throw NotFoundException(Error(s"registry $registryName not found"))
}
}
@POST
@Path("/{registryName}/schemas/{schema}/sql-avatica")
//@Produces(Array(MediaType.APPLICATION_JSON))
//@Consumes(Array(MediaType.APPLICATION_JSON))
def avatica(@PathParam("registryName") registryName: String,
@PathParam("schema") schema: String,
@QueryParam("debug") @DefaultValue("false") requestDebug: Boolean,
@Context httpServletRequest: HttpServletRequest,
@Context containerRequestContext: ContainerRequestContext,
@Suspended response: AsyncResponse) : Unit = {
val serialization = httpServletRequest.getParameter("serialization")
serialization match {
case "PROTOBUF" | "protobuf" =>
val translationer = new ProtobufTranslationImpl
val avaticaProtobufHandler = new AvaticaMahaProtobufHandler(mahaAvaticaService, translationer, NoopMetricsSystem.getInstance())
val wireMessage = WireMessage.parseFrom(httpServletRequest.getInputStream)
logger.info(s"wireMessage type: ${wireMessage.getName}")
val handlerResponse = avaticaProtobufHandler.apply(wireMessage.toByteArray)
response.resume(handlerResponse.getResponse)
case _ =>
val avaticaJsonHandler = new AvaticaMahaJsonHandler(mahaAvaticaService, NoopMetricsSystem.getInstance())
val rawString = IOUtils.toByteArray(httpServletRequest.getInputStream)
val json = new String(rawString)
logger.info(s"parsed json: " + json)
val handlerResponse = avaticaJsonHandler.apply(json)
logger.info(handlerResponse.getResponse)
response.resume(handlerResponse.getResponse)
}
}
@POST
@Path("/{registryName}/schemas/{schema}/query")
@Produces(Array(MediaType.APPLICATION_JSON))
@Consumes(Array(MediaType.APPLICATION_JSON))
def query(@PathParam("registryName") registryName: String,
@PathParam("schema") schema: String,
@QueryParam("debug") @DefaultValue("false") requestDebug: Boolean,
@QueryParam("forceEngine") forceEngine: String,
@QueryParam("forceRevision") forceRevision: Int,
@QueryParam("testName") testName: String,
@QueryParam("labels") labels: java.util.List[String],
@Context httpServletRequest: HttpServletRequest,
@Context containerRequestContext: ContainerRequestContext,
@Suspended response: AsyncResponse) : Unit = {
val requestStartTime = System.currentTimeMillis()
info(s"registryName: $registryName, schema: $schema, forceEngine: $forceEngine, forceRevision: $forceRevision")
val schemaOption: Option[Schema] = Schema.withNameInsensitiveOption(schema)
if(schemaOption.isEmpty) {
throw NotFoundException(Error(s"schema $schema not found"))
}
val userId: String = Option(MDC.get(MahaConstants.USER_ID)).getOrElse("unknown")
val requestId: String = Option(MDC.get(MahaConstants.REQUEST_ID)).getOrElse(UUID.randomUUID().toString)
val debug: Boolean = if(requestDebug) requestDebug else debugUsers(userId)
val (reportingRequest: ReportingRequest, rawJson: Array[Byte]) = createReportingRequest(
requestId
, userId
, httpServletRequest
, schemaOption.get
, debug
, forceEngine
, testName
, labels
)
val bucketParams: BucketParams = BucketParams(UserInfo(userId, Try(MDC.get(MahaConstants.IS_INTERNAL).toBoolean).getOrElse(false)), forceRevision = Option(forceRevision))
val mahaRequestContext: MahaRequestContext = mahaRequestContextBuilder.build(registryName
, bucketParams, reportingRequest, rawJson, requestId, userId, requestStartTime = requestStartTime, containerRequestContext)
requestValidator.validate(mahaRequestContext, containerRequestContext)
val mahaRequestProcessor: MahaSyncRequestProcessor = mahaRequestProcessorFactory
.create(mahaRequestContext, MahaServiceConstants.MahaRequestLabel)
mahaRequestProcessor.onSuccess((requestCoordinatorResult: RequestCoordinatorResult) => {
response.resume(new JsonStreamingOutput(requestCoordinatorResult, debugRenderer = defaultDebugRenderer))
})
mahaRequestProcessor.onFailure((ge: GeneralError) => {
if(ge.throwableOption.isDefined) {
val error = ge.throwableOption.get
this.error(ge.message, error)
if(error.getCause != null) {
response.resume(error.getCause)
} else {
response.resume(error)
}
} else {
response.resume(new Exception(ge.message))
}
})
mahaRequestProcessor.process()
}
private def createReportingRequest(requestId:String
, userId:String
, httpServletRequest: HttpServletRequest
, schema: Schema
, debug: Boolean
, forceEngine: String
, testName: String
, labels: java.util.List[String]) : (ReportingRequest, Array[Byte]) = {
val rawJson = IOUtils.toByteArray(httpServletRequest.getInputStream)
val reportingRequestResult = baseRequest.deserializeSyncWithFactBias(rawJson, schema)
require(reportingRequestResult.isSuccess, reportingRequestResult.toString)
val originalRequest = reportingRequestResult.toOption.get
val request = {
if(!debug && StringUtils.isBlank(forceEngine)) {
originalRequest
} else {
val withDebug = if(debug) {
ReportingRequest.enableDebug(originalRequest)
} else {
originalRequest
}
val withEngine = if(StringUtils.isNotBlank(forceEngine)) {
Engine.from(forceEngine).fold(withDebug) {
case OracleEngine => ReportingRequest.forceOracle(withDebug)
case DruidEngine => ReportingRequest.forceDruid(withDebug)
case HiveEngine => ReportingRequest.forceHive(withDebug)
case PrestoEngine => ReportingRequest.forcePresto(withDebug)
case PostgresEngine => ReportingRequest.forcePostgres(withDebug)
case BigqueryEngine => ReportingRequest.forceBigquery(withDebug)
case _ => withDebug
}
} else {
withDebug
}
val withTestName = if(StringUtils.isNotBlank(testName)) {
ReportingRequest.withTestName(withEngine, testName)
} else {
withEngine
}
val withLabels = if(labels != null && !labels.isEmpty) {
import scala.collection.JavaConverters._
ReportingRequest.withLabels(withTestName, labels.asScala.toList)
} else {
withTestName
}
withLabels
}
}
(ReportingRequest.addRequestContext(request, RequestContext(requestId, userId)), rawJson)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy