mirror of
https://github.com/twitter/the-algorithm.git
synced 2025-01-12 12:19:10 +01:00
[docx] split commit for file 3200
Signed-off-by: Ari Archer <ari.web.xyz@gmail.com>
This commit is contained in:
parent
23cec12533
commit
f3c5ff35cb
Binary file not shown.
@ -1,57 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline
|
||||
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.MalformedCursor
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.scrooge.BinaryThriftStructSerializer
|
||||
import com.twitter.scrooge.ThriftStruct
|
||||
import com.twitter.util.Return
|
||||
import com.twitter.util.Throw
|
||||
import com.twitter.util.Try
|
||||
|
||||
/**
|
||||
* Serializes a [[PipelineCursor]] into thrift and then into a base64 encoded string
|
||||
*/
|
||||
trait PipelineCursorSerializer[-Cursor <: PipelineCursor] {
|
||||
def serializeCursor(cursor: Cursor): String
|
||||
}
|
||||
|
||||
object PipelineCursorSerializer {
|
||||
|
||||
/**
|
||||
* Deserializes a cursor string into thrift and then into a [[PipelineCursor]]
|
||||
*
|
||||
* @param cursorString to deserialize, which is base64 encoded thrift
|
||||
* @param cursorThriftSerializer to deserialize the cursor string into thrift
|
||||
* @param deserializePf specifies how to transform the serialized thrift into a [[PipelineCursor]]
|
||||
* @return optional [[PipelineCursor]]. `None` may or may not be a failure depending on the
|
||||
* implementation of deserializePf.
|
||||
*
|
||||
* @note The "A" type of deserializePf cannot be inferred due to the thrift type not being present
|
||||
* on the PipelineCursorSerializer trait. Therefore invokers must often add an explicit type
|
||||
* on the deserializeCursor call to help out the compiler when passing deserializePf inline.
|
||||
* Alternatively, deserializePf can be declared as a val with a type annotation before it is
|
||||
* passed into this method.
|
||||
*/
|
||||
def deserializeCursor[Thrift <: ThriftStruct, Cursor <: PipelineCursor](
|
||||
cursorString: String,
|
||||
cursorThriftSerializer: BinaryThriftStructSerializer[Thrift],
|
||||
deserializePf: PartialFunction[Option[Thrift], Option[Cursor]]
|
||||
): Option[Cursor] = {
|
||||
val thriftCursor: Option[Thrift] =
|
||||
Try {
|
||||
cursorThriftSerializer.fromString(cursorString)
|
||||
} match {
|
||||
case Return(thriftCursor) => Some(thriftCursor)
|
||||
case Throw(_) => None
|
||||
}
|
||||
|
||||
// Add type annotation to help out the compiler since the type is lost due to the _ match
|
||||
val defaultDeserializePf: PartialFunction[Option[Thrift], Option[Cursor]] = {
|
||||
case _ =>
|
||||
// This case is the result of the client submitting a cursor we do not expect
|
||||
throw PipelineFailure(MalformedCursor, s"Unknown request cursor: $cursorString")
|
||||
}
|
||||
|
||||
(deserializePf orElse defaultDeserializePf)(thriftCursor)
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,32 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline
|
||||
|
||||
import com.twitter.product_mixer.core.feature.featuremap.FeatureMap
|
||||
import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext
|
||||
import com.twitter.product_mixer.core.model.marshalling.request.HasDebugOptions
|
||||
import com.twitter.product_mixer.core.model.marshalling.request.HasProduct
|
||||
import com.twitter.timelines.configapi.HasParams
|
||||
import com.twitter.timelines.configapi.Param
|
||||
import com.twitter.util.Time
|
||||
|
||||
trait PipelineQuery extends HasParams with HasClientContext with HasProduct with HasDebugOptions {
|
||||
self =>
|
||||
|
||||
/** Set a query time val that is constant for the duration of the query lifecycle */
|
||||
val queryTime: Time = self.debugOptions.flatMap(_.requestTimeOverride).getOrElse(Time.now)
|
||||
|
||||
/** The requested max results is specified, or not specified, by the thrift client */
|
||||
def requestedMaxResults: Option[Int]
|
||||
|
||||
/** Retrieves the max results with a default Param, if not specified by the thrift client */
|
||||
def maxResults(defaultRequestedMaxResultParam: Param[Int]): Int =
|
||||
requestedMaxResults.getOrElse(params(defaultRequestedMaxResultParam))
|
||||
|
||||
/** Optional [[FeatureMap]], this may be updated later using [[withFeatureMap]] */
|
||||
def features: Option[FeatureMap]
|
||||
|
||||
/**
|
||||
* Since Query-Level features can be hydrated later, we need this method to update the PipelineQuery
|
||||
* usually this will be implemented via `copy(features = Some(features))`
|
||||
*/
|
||||
def withFeatureMap(features: FeatureMap): PipelineQuery
|
||||
}
|
Binary file not shown.
@ -1,59 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline
|
||||
|
||||
import com.twitter.product_mixer.component_library.model.candidate.CursorCandidate
|
||||
import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails
|
||||
import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails
|
||||
import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.ExecutionFailed
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.util.Return
|
||||
import com.twitter.util.Throw
|
||||
import com.twitter.util.Try
|
||||
|
||||
/**
|
||||
* Pipelines return a PipelineResult.
|
||||
*
|
||||
* This allows us to return a single main result (optionally, incase the pipeline didn't execute successfully), but
|
||||
* still have a detailed response object to show how that result was produced.
|
||||
*/
|
||||
trait PipelineResult[ResultType] {
|
||||
val failure: Option[PipelineFailure]
|
||||
val result: Option[ResultType]
|
||||
|
||||
def withFailure(failure: PipelineFailure): PipelineResult[ResultType]
|
||||
def withResult(result: ResultType): PipelineResult[ResultType]
|
||||
|
||||
def resultSize(): Int
|
||||
|
||||
private[pipeline] def stopExecuting: Boolean = failure.isDefined || result.isDefined
|
||||
|
||||
final def toTry: Try[this.type] = (result, failure) match {
|
||||
case (_, Some(failure)) =>
|
||||
Throw(failure)
|
||||
case (_: Some[ResultType], _) =>
|
||||
Return(this)
|
||||
// Pipelines should always finish with either a result or a failure
|
||||
case _ => Throw(PipelineFailure(ExecutionFailed, "Pipeline did not execute"))
|
||||
}
|
||||
|
||||
final def toResultTry: Try[ResultType] = {
|
||||
// `.get` is safe here because `toTry` guarantees a value in the `Return` case
|
||||
toTry.map(_.result.get)
|
||||
}
|
||||
}
|
||||
|
||||
object PipelineResult {
|
||||
|
||||
/**
|
||||
* Track number of candidates returned by a Pipeline. Cursors are excluded from this
|
||||
* count and modules are counted as the sum of their candidates.
|
||||
*
|
||||
* @note this is a somewhat subjective measure of 'size' and it is spread across pipeline
|
||||
* definitions as well as selectors.
|
||||
*/
|
||||
def resultSize(results: Seq[CandidateWithDetails]): Int = results.map {
|
||||
case module: ModuleCandidateWithDetails => resultSize(module.candidates)
|
||||
case ItemCandidateWithDetails(_: CursorCandidate, _, _) => 0
|
||||
case _ => 1
|
||||
}.sum
|
||||
}
|
@ -1,77 +0,0 @@
|
||||
scala_library(
|
||||
sources = ["*.scala"],
|
||||
compiler_option_sets = ["fatal_warnings"],
|
||||
platform = "java8",
|
||||
strict_deps = True,
|
||||
tags = ["bazel-compatible"],
|
||||
dependencies = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"configapi/configapi-core",
|
||||
"configapi/configapi-decider",
|
||||
"finatra/inject/inject-core/src/main/scala",
|
||||
"finatra/inject/inject-core/src/main/scala/com/twitter/inject",
|
||||
"product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featurestorev1",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/async_feature_map_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_decorator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_feature_hydrator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_source_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/filter_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/group_results_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util",
|
||||
"stitch/stitch-core",
|
||||
"util/util-core:scala",
|
||||
],
|
||||
exports = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"configapi/configapi-core",
|
||||
"configapi/configapi-decider",
|
||||
"finatra/inject/inject-core/src/main/scala",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/async_feature_map_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_decorator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_feature_hydrator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_source_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/filter_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/group_results_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor",
|
||||
"stitch/stitch-core",
|
||||
],
|
||||
)
|
Binary file not shown.
Binary file not shown.
@ -1,30 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.candidate
|
||||
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails
|
||||
import com.twitter.product_mixer.core.pipeline.Pipeline
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.stitch.Arrow
|
||||
|
||||
/**
|
||||
* A Candidate Pipeline
|
||||
*
|
||||
* This is an abstract class, as we only construct these via the [[CandidatePipelineBuilder]].
|
||||
*
|
||||
* A [[CandidatePipeline]] is capable of processing requests (queries) and returning candidates
|
||||
* in the form of a [[CandidatePipelineResult]]
|
||||
*
|
||||
* @tparam Query the domain model for the query or request
|
||||
*/
|
||||
abstract class CandidatePipeline[-Query <: PipelineQuery] private[candidate]
|
||||
extends Pipeline[CandidatePipeline.Inputs[Query], Seq[CandidateWithDetails]] {
|
||||
override private[core] val config: BaseCandidatePipelineConfig[Query, _, _, _]
|
||||
override val arrow: Arrow[CandidatePipeline.Inputs[Query], CandidatePipelineResult]
|
||||
override val identifier: CandidatePipelineIdentifier
|
||||
}
|
||||
|
||||
object CandidatePipeline {
|
||||
case class Inputs[+Query <: PipelineQuery](
|
||||
query: Query,
|
||||
existingCandidates: Seq[CandidateWithDetails])
|
||||
}
|
Binary file not shown.
@ -1,735 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.candidate
|
||||
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.product_mixer.core.feature.featuremap.FeatureMap
|
||||
import com.twitter.product_mixer.core.feature.featuremap.asyncfeaturemap.AsyncFeatureMap
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator
|
||||
import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseQueryFeatureHydrator
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.BaseCandidatePipelineQueryTransformer
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer
|
||||
import com.twitter.product_mixer.core.gate.ParamGate
|
||||
import com.twitter.product_mixer.core.gate.ParamGate.EnabledGateSuffix
|
||||
import com.twitter.product_mixer.core.gate.ParamGate.SupportedClientGateSuffix
|
||||
import com.twitter.product_mixer.core.model.common.CandidateWithFeatures
|
||||
import com.twitter.product_mixer.core.model.common.Component
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails
|
||||
import com.twitter.product_mixer.core.pipeline.InvalidStepStateException
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineBuilder
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.ClosedGate
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailureClassifier
|
||||
import com.twitter.product_mixer.core.service.Executor
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutor
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutorResults
|
||||
import com.twitter.product_mixer.core.service.candidate_decorator_executor.CandidateDecoratorExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_decorator_executor.CandidateDecoratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.candidate_source_executor.CandidateSourceExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_source_executor.CandidateSourceExecutorResult
|
||||
import com.twitter.product_mixer.core.service.candidate_source_executor.FetchedCandidateWithFeatures
|
||||
import com.twitter.product_mixer.core.service.filter_executor.FilterExecutor
|
||||
import com.twitter.product_mixer.core.service.filter_executor.FilterExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.StoppedGateException
|
||||
import com.twitter.product_mixer.core.service.group_results_executor.GroupResultsExecutor
|
||||
import com.twitter.product_mixer.core.service.group_results_executor.GroupResultsExecutorInput
|
||||
import com.twitter.product_mixer.core.service.group_results_executor.GroupResultsExecutorResult
|
||||
import com.twitter.product_mixer.core.service.query_feature_hydrator_executor.QueryFeatureHydratorExecutor
|
||||
import com.twitter.stitch.Arrow
|
||||
import com.twitter.util.logging.Logging
|
||||
import javax.inject.Inject
|
||||
|
||||
class CandidatePipelineBuilder[
|
||||
Query <: PipelineQuery,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result <: UniversalNoun[Any]] @Inject() (
|
||||
queryFeatureHydratorExecutor: QueryFeatureHydratorExecutor,
|
||||
asyncFeatureMapExecutor: AsyncFeatureMapExecutor,
|
||||
candidateDecoratorExecutor: CandidateDecoratorExecutor,
|
||||
candidateFeatureHydratorExecutor: CandidateFeatureHydratorExecutor,
|
||||
candidateSourceExecutor: CandidateSourceExecutor,
|
||||
groupResultsExecutor: GroupResultsExecutor,
|
||||
filterExecutor: FilterExecutor,
|
||||
gateExecutor: GateExecutor,
|
||||
override val statsReceiver: StatsReceiver)
|
||||
extends PipelineBuilder[CandidatePipeline.Inputs[Query]]
|
||||
with Logging {
|
||||
|
||||
override type UnderlyingResultType = Seq[CandidateWithDetails]
|
||||
override type PipelineResultType = IntermediateCandidatePipelineResult[Result]
|
||||
|
||||
def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
config: BaseCandidatePipelineConfig[
|
||||
Query,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result
|
||||
]
|
||||
): CandidatePipeline[Query] = {
|
||||
|
||||
val pipelineIdentifier = config.identifier
|
||||
val candidateSourceIdentifier = config.candidateSource.identifier
|
||||
|
||||
val context = Executor.Context(
|
||||
PipelineFailureClassifier(
|
||||
config.failureClassifier.orElse(StoppedGateException.classifier(ClosedGate))),
|
||||
parentComponentIdentifierStack.push(pipelineIdentifier)
|
||||
)
|
||||
|
||||
val enabledGateOpt = config.enabledDeciderParam.map { deciderParam =>
|
||||
ParamGate(pipelineIdentifier + EnabledGateSuffix, deciderParam)
|
||||
}
|
||||
val supportedClientGateOpt = config.supportedClientParam.map { param =>
|
||||
ParamGate(pipelineIdentifier + SupportedClientGateSuffix, param)
|
||||
}
|
||||
|
||||
/**
|
||||
* Evaluate enabled decider gate first since if it's off, there is no reason to proceed
|
||||
* Next evaluate supported client feature switch gate, followed by customer configured gates
|
||||
*/
|
||||
val allGates = enabledGateOpt.toSeq ++ supportedClientGateOpt.toSeq ++ config.gates
|
||||
|
||||
// Dynamically replace the identifier of both transformers if config used the inline constructor
|
||||
// which sets a default identifier. We need to do this to ensure uniqueness of identifiers.
|
||||
val queryTransformer = BaseCandidatePipelineQueryTransformer.copyWithUpdatedIdentifier(
|
||||
config.queryTransformer,
|
||||
pipelineIdentifier)
|
||||
|
||||
val resultsTransformer = CandidatePipelineResultsTransformer.copyWithUpdatedIdentifier(
|
||||
config.resultTransformer,
|
||||
pipelineIdentifier)
|
||||
|
||||
val decorator = config.decorator.map(decorator =>
|
||||
CandidateDecorator.copyWithUpdatedIdentifier(decorator, pipelineIdentifier))
|
||||
|
||||
val GatesStep = new Step[Query, GateExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = CandidatePipelineConfig.gatesStep
|
||||
|
||||
override def executorArrow: Arrow[Query, GateExecutorResult] = {
|
||||
gateExecutor.arrow(allGates, context)
|
||||
}
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): Query =
|
||||
query.query
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: GateExecutorResult
|
||||
): IntermediateCandidatePipelineResult[Result] =
|
||||
previousPipelineResult.copy(underlyingResult =
|
||||
previousPipelineResult.underlyingResult.copy(gateResult = Some(executorResult)))
|
||||
}
|
||||
|
||||
def queryFeatureHydrationStep(
|
||||
queryFeatureHydrators: Seq[BaseQueryFeatureHydrator[Query, _]],
|
||||
stepIdentifier: PipelineStepIdentifier,
|
||||
updater: ResultUpdater[CandidatePipelineResult, QueryFeatureHydratorExecutor.Result]
|
||||
): Step[Query, QueryFeatureHydratorExecutor.Result] =
|
||||
new Step[Query, QueryFeatureHydratorExecutor.Result] {
|
||||
override def identifier: PipelineStepIdentifier = stepIdentifier
|
||||
|
||||
override def executorArrow: Arrow[Query, QueryFeatureHydratorExecutor.Result] =
|
||||
queryFeatureHydratorExecutor.arrow(
|
||||
queryFeatureHydrators,
|
||||
CandidatePipelineConfig.stepsAsyncFeatureHydrationCanBeCompletedBy,
|
||||
context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): Query = query.query
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: QueryFeatureHydratorExecutor.Result
|
||||
): IntermediateCandidatePipelineResult[Result] =
|
||||
previousPipelineResult.copy(
|
||||
underlyingResult = updater(previousPipelineResult.underlyingResult, executorResult))
|
||||
|
||||
override def queryUpdater(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
executorResult: QueryFeatureHydratorExecutor.Result
|
||||
): CandidatePipeline.Inputs[Query] =
|
||||
CandidatePipeline.Inputs(
|
||||
query.query
|
||||
.withFeatureMap(
|
||||
query.query.features.getOrElse(
|
||||
FeatureMap.empty) ++ executorResult.featureMap).asInstanceOf[Query],
|
||||
query.existingCandidates)
|
||||
}
|
||||
|
||||
def asyncFeaturesStep(
|
||||
stepToHydrateFor: PipelineStepIdentifier,
|
||||
context: Executor.Context
|
||||
): Step[AsyncFeatureMap, AsyncFeatureMapExecutorResults] =
|
||||
new Step[AsyncFeatureMap, AsyncFeatureMapExecutorResults] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
CandidatePipelineConfig.asyncFeaturesStep(stepToHydrateFor)
|
||||
|
||||
override def executorArrow: Arrow[AsyncFeatureMap, AsyncFeatureMapExecutorResults] =
|
||||
asyncFeatureMapExecutor.arrow(stepToHydrateFor, identifier, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): AsyncFeatureMap =
|
||||
previousResult.underlyingResult.mergedAsyncQueryFeatures
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "MergedAsyncQueryFeatures")
|
||||
)
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: AsyncFeatureMapExecutorResults
|
||||
): IntermediateCandidatePipelineResult[Result] =
|
||||
previousPipelineResult.copy(
|
||||
underlyingResult =
|
||||
previousPipelineResult.underlyingResult.copy(asyncFeatureHydrationResults =
|
||||
previousPipelineResult.underlyingResult.asyncFeatureHydrationResults match {
|
||||
case Some(existingResults) => Some(existingResults ++ executorResult)
|
||||
case None => Some(executorResult)
|
||||
}))
|
||||
|
||||
override def queryUpdater(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
executorResult: AsyncFeatureMapExecutorResults
|
||||
): CandidatePipeline.Inputs[Query] =
|
||||
if (executorResult.featureMapsByStep
|
||||
.getOrElse(stepToHydrateFor, FeatureMap.empty).isEmpty) {
|
||||
query
|
||||
} else {
|
||||
val updatedQuery = query.query
|
||||
.withFeatureMap(
|
||||
query.query.features
|
||||
.getOrElse(FeatureMap.empty) ++ executorResult.featureMapsByStep(
|
||||
stepToHydrateFor)).asInstanceOf[Query]
|
||||
CandidatePipeline.Inputs(updatedQuery, query.existingCandidates)
|
||||
}
|
||||
}
|
||||
|
||||
val CandidateSourceStep =
|
||||
new Step[Query, CandidateSourceExecutorResult[Result]] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
CandidatePipelineConfig.candidateSourceStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
Query,
|
||||
CandidateSourceExecutorResult[Result]
|
||||
] =
|
||||
candidateSourceExecutor
|
||||
.arrow(
|
||||
config.candidateSource,
|
||||
queryTransformer,
|
||||
resultsTransformer,
|
||||
config.featuresFromCandidateSourceTransformers,
|
||||
context
|
||||
)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): Query =
|
||||
query.query
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: CandidateSourceExecutorResult[Result]
|
||||
): IntermediateCandidatePipelineResult[Result] =
|
||||
previousPipelineResult.copy(underlyingResult =
|
||||
previousPipelineResult.underlyingResult.copy(
|
||||
candidateSourceResult =
|
||||
Some(executorResult.asInstanceOf[CandidateSourceExecutorResult[UniversalNoun[Any]]])
|
||||
))
|
||||
|
||||
override def queryUpdater(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
executorResult: CandidateSourceExecutorResult[Result]
|
||||
): CandidatePipeline.Inputs[Query] = {
|
||||
val updatedFeatureMap =
|
||||
query.query.features
|
||||
.getOrElse(FeatureMap.empty) ++ executorResult.candidateSourceFeatureMap
|
||||
val updatedQuery = query.query
|
||||
.withFeatureMap(updatedFeatureMap).asInstanceOf[Query]
|
||||
CandidatePipeline.Inputs(updatedQuery, query.existingCandidates)
|
||||
}
|
||||
}
|
||||
|
||||
val PreFilterFeatureHydrationPhase1Step =
|
||||
new Step[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Result],
|
||||
CandidateFeatureHydratorExecutorResult[Result]
|
||||
] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
CandidatePipelineConfig.preFilterFeatureHydrationPhase1Step
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Result],
|
||||
CandidateFeatureHydratorExecutorResult[Result]
|
||||
] =
|
||||
candidateFeatureHydratorExecutor.arrow(config.preFilterFeatureHydrationPhase1, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): CandidateFeatureHydratorExecutor.Inputs[Query, Result] = {
|
||||
val candidateSourceExecutorResult =
|
||||
previousResult.underlyingResult.candidateSourceResult.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "CandidateSourceResult")
|
||||
}
|
||||
CandidateFeatureHydratorExecutor.Inputs(
|
||||
query.query,
|
||||
candidateSourceExecutorResult.candidates
|
||||
.asInstanceOf[Seq[CandidateWithFeatures[Result]]])
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: CandidateFeatureHydratorExecutorResult[Result]
|
||||
): IntermediateCandidatePipelineResult[Result] = {
|
||||
val candidateSourceExecutorResult =
|
||||
previousPipelineResult.underlyingResult.candidateSourceResult.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "CandidateSourceResult")
|
||||
}
|
||||
|
||||
val featureMapsFromPreFilter = executorResult.results.map { result =>
|
||||
result.candidate -> result.features
|
||||
}.toMap
|
||||
|
||||
val mergedFeatureMaps = candidateSourceExecutorResult.candidates.map { candidate =>
|
||||
val candidateFeatureMap = candidate.features
|
||||
val preFilterFeatureMap =
|
||||
featureMapsFromPreFilter.getOrElse(
|
||||
candidate.candidate.asInstanceOf[Result],
|
||||
FeatureMap.empty)
|
||||
|
||||
candidate.candidate.asInstanceOf[Result] -> (candidateFeatureMap ++ preFilterFeatureMap)
|
||||
}.toMap
|
||||
|
||||
previousPipelineResult.copy(
|
||||
underlyingResult = previousPipelineResult.underlyingResult.copy(
|
||||
preFilterHydrationResult = Some(
|
||||
executorResult
|
||||
.asInstanceOf[CandidateFeatureHydratorExecutorResult[UniversalNoun[Any]]])
|
||||
),
|
||||
featureMaps = Some(mergedFeatureMaps)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
val PreFilterFeatureHydrationPhase2Step =
|
||||
new Step[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Result],
|
||||
CandidateFeatureHydratorExecutorResult[Result]
|
||||
] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
CandidatePipelineConfig.preFilterFeatureHydrationPhase2Step
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Result],
|
||||
CandidateFeatureHydratorExecutorResult[Result]
|
||||
] =
|
||||
candidateFeatureHydratorExecutor.arrow(config.preFilterFeatureHydrationPhase2, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): CandidateFeatureHydratorExecutor.Inputs[Query, Result] = {
|
||||
val candidates = previousResult.underlyingResult.preFilterHydrationResult.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "PreFilterHydrationResult")
|
||||
}.results
|
||||
CandidateFeatureHydratorExecutor.Inputs(
|
||||
query.query,
|
||||
candidates.asInstanceOf[Seq[CandidateWithFeatures[Result]]]
|
||||
)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: CandidateFeatureHydratorExecutorResult[Result]
|
||||
): IntermediateCandidatePipelineResult[Result] = {
|
||||
|
||||
val featureMapsFromPreFilterPhase2 = executorResult.results.map { result =>
|
||||
result.candidate -> result.features
|
||||
}.toMap
|
||||
|
||||
val mergedFeatureMaps = previousPipelineResult.featureMaps
|
||||
.getOrElse(throw InvalidStepStateException(identifier, "FeatureMaps"))
|
||||
.map {
|
||||
case (candidate, featureMap) =>
|
||||
val preFilterPhase2FeatureMap =
|
||||
featureMapsFromPreFilterPhase2.getOrElse(candidate, FeatureMap.empty)
|
||||
|
||||
candidate -> (featureMap ++ preFilterPhase2FeatureMap)
|
||||
}
|
||||
|
||||
previousPipelineResult.copy(
|
||||
underlyingResult = previousPipelineResult.underlyingResult.copy(
|
||||
preFilterHydrationResultPhase2 = Some(
|
||||
executorResult
|
||||
.asInstanceOf[CandidateFeatureHydratorExecutorResult[UniversalNoun[Any]]])
|
||||
),
|
||||
featureMaps = Some(mergedFeatureMaps)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
val FiltersStep =
|
||||
new Step[(Query, Seq[CandidateWithFeatures[Result]]), FilterExecutorResult[Result]] {
|
||||
override def identifier: PipelineStepIdentifier = CandidatePipelineConfig.filtersStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
(Query, Seq[CandidateWithFeatures[Result]]),
|
||||
FilterExecutorResult[
|
||||
Result
|
||||
]
|
||||
] =
|
||||
filterExecutor.arrow(config.filters, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): (Query, Seq[CandidateWithFeatures[Result]]) = {
|
||||
val candidates =
|
||||
previousResult.underlyingResult.candidateSourceResult
|
||||
.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "CandidateSourceResult")
|
||||
}.candidates.map(_.candidate).asInstanceOf[Seq[Result]]
|
||||
|
||||
val featureMaps = previousResult.featureMaps
|
||||
.getOrElse(throw InvalidStepStateException(identifier, "FeatureMaps"))
|
||||
|
||||
(
|
||||
query.query,
|
||||
candidates.map(candidate =>
|
||||
CandidateWithFeaturesImpl(
|
||||
candidate,
|
||||
featureMaps.getOrElse(candidate, FeatureMap.empty))))
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: FilterExecutorResult[Result]
|
||||
): IntermediateCandidatePipelineResult[Result] =
|
||||
previousPipelineResult.copy(underlyingResult =
|
||||
previousPipelineResult.underlyingResult.copy(
|
||||
filterResult =
|
||||
Some(executorResult.asInstanceOf[FilterExecutorResult[UniversalNoun[Any]]])
|
||||
))
|
||||
}
|
||||
|
||||
val PostFilterFeatureHydrationStep =
|
||||
new Step[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Result],
|
||||
CandidateFeatureHydratorExecutorResult[Result]
|
||||
] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
CandidatePipelineConfig.postFilterFeatureHydrationStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Result],
|
||||
CandidateFeatureHydratorExecutorResult[Result]
|
||||
] =
|
||||
candidateFeatureHydratorExecutor.arrow(config.postFilterFeatureHydration, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): CandidateFeatureHydratorExecutor.Inputs[Query, Result] = {
|
||||
val filterResult = previousResult.underlyingResult.filterResult
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FilterResult")
|
||||
).result.asInstanceOf[Seq[Result]]
|
||||
|
||||
val featureMaps = previousResult.featureMaps.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FeatureMaps")
|
||||
)
|
||||
|
||||
val filteredCandidates = filterResult.map { candidate =>
|
||||
CandidateWithFeaturesImpl(candidate, featureMaps.getOrElse(candidate, FeatureMap.empty))
|
||||
}
|
||||
CandidateFeatureHydratorExecutor.Inputs(query.query, filteredCandidates)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: CandidateFeatureHydratorExecutorResult[Result]
|
||||
): IntermediateCandidatePipelineResult[Result] = {
|
||||
val filterResult = previousPipelineResult.underlyingResult.filterResult
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FilterResult")
|
||||
).result.asInstanceOf[Seq[Result]]
|
||||
|
||||
val featureMaps = previousPipelineResult.featureMaps.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FeatureMaps")
|
||||
)
|
||||
|
||||
val postFilterFeatureMaps = executorResult.results.map { result =>
|
||||
result.candidate -> result.features
|
||||
}.toMap
|
||||
|
||||
val mergedFeatureMaps = filterResult.map { candidate =>
|
||||
candidate ->
|
||||
(featureMaps
|
||||
.getOrElse(candidate, FeatureMap.empty) ++ postFilterFeatureMaps.getOrElse(
|
||||
candidate,
|
||||
FeatureMap.empty))
|
||||
}.toMap
|
||||
|
||||
previousPipelineResult.copy(
|
||||
underlyingResult = previousPipelineResult.underlyingResult.copy(
|
||||
postFilterHydrationResult = Some(
|
||||
executorResult
|
||||
.asInstanceOf[CandidateFeatureHydratorExecutorResult[UniversalNoun[Any]]])
|
||||
),
|
||||
featureMaps = Some(mergedFeatureMaps)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
val ScorersStep =
|
||||
new Step[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Result],
|
||||
CandidateFeatureHydratorExecutorResult[Result]
|
||||
] {
|
||||
override def identifier: PipelineStepIdentifier = CandidatePipelineConfig.scorersStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Result],
|
||||
CandidateFeatureHydratorExecutorResult[Result]
|
||||
] =
|
||||
candidateFeatureHydratorExecutor.arrow(config.scorers, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): CandidateFeatureHydratorExecutor.Inputs[Query, Result] = {
|
||||
val filterResult = previousResult.underlyingResult.filterResult
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FilterResult")
|
||||
).result.asInstanceOf[Seq[Result]]
|
||||
|
||||
val featureMaps = previousResult.featureMaps.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FeatureMaps")
|
||||
)
|
||||
|
||||
val filteredCandidates = filterResult.map { candidate =>
|
||||
CandidateWithFeaturesImpl(candidate, featureMaps.getOrElse(candidate, FeatureMap.empty))
|
||||
}
|
||||
CandidateFeatureHydratorExecutor.Inputs(query.query, filteredCandidates)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: CandidateFeatureHydratorExecutorResult[Result]
|
||||
): IntermediateCandidatePipelineResult[Result] = {
|
||||
val filterResult = previousPipelineResult.underlyingResult.filterResult
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FilterResult")
|
||||
).result.asInstanceOf[Seq[Result]]
|
||||
|
||||
val featureMaps = previousPipelineResult.featureMaps.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FeatureMaps")
|
||||
)
|
||||
|
||||
val scoringFeatureMaps = executorResult.results.map { result =>
|
||||
result.candidate -> result.features
|
||||
}.toMap
|
||||
|
||||
val mergedFeatureMaps = filterResult.map { candidate =>
|
||||
candidate ->
|
||||
(featureMaps
|
||||
.getOrElse(candidate, FeatureMap.empty) ++ scoringFeatureMaps.getOrElse(
|
||||
candidate,
|
||||
FeatureMap.empty))
|
||||
}.toMap
|
||||
|
||||
previousPipelineResult.copy(
|
||||
underlyingResult = previousPipelineResult.underlyingResult.copy(
|
||||
scorersResult = Some(
|
||||
executorResult
|
||||
.asInstanceOf[CandidateFeatureHydratorExecutorResult[UniversalNoun[Any]]])
|
||||
),
|
||||
featureMaps = Some(mergedFeatureMaps)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
val DecorationStep =
|
||||
new Step[(Query, Seq[CandidateWithFeatures[Result]]), CandidateDecoratorExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = CandidatePipelineConfig.decoratorStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
(Query, Seq[CandidateWithFeatures[Result]]),
|
||||
CandidateDecoratorExecutorResult
|
||||
] =
|
||||
candidateDecoratorExecutor.arrow(decorator, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): (Query, Seq[CandidateWithFeatures[Result]]) = {
|
||||
val keptCandidates = previousResult.underlyingResult.filterResult
|
||||
.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "FilterResult")
|
||||
}.result.asInstanceOf[Seq[Result]]
|
||||
|
||||
val featureMaps = previousResult.featureMaps.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "FeatureMaps")
|
||||
}
|
||||
|
||||
(
|
||||
query.query,
|
||||
keptCandidates.map(candidate =>
|
||||
CandidateWithFeaturesImpl(
|
||||
candidate,
|
||||
featureMaps.getOrElse(candidate, FeatureMap.empty))))
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: CandidateDecoratorExecutorResult
|
||||
): IntermediateCandidatePipelineResult[Result] =
|
||||
previousPipelineResult.copy(underlyingResult =
|
||||
previousPipelineResult.underlyingResult.copy(
|
||||
candidateDecoratorResult = Some(executorResult)
|
||||
))
|
||||
}
|
||||
|
||||
/**
|
||||
* ResultStep is a synchronous step that basically takes the outputs from the other steps, groups modules,
|
||||
* and puts things into the final result object
|
||||
*/
|
||||
val ResultStep = new Step[GroupResultsExecutorInput[Result], GroupResultsExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = CandidatePipelineConfig.resultStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
GroupResultsExecutorInput[Result],
|
||||
GroupResultsExecutorResult
|
||||
] = groupResultsExecutor.arrow(pipelineIdentifier, candidateSourceIdentifier, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: CandidatePipeline.Inputs[Query],
|
||||
previousResult: IntermediateCandidatePipelineResult[Result]
|
||||
): GroupResultsExecutorInput[Result] = {
|
||||
|
||||
val underlying = previousResult.underlyingResult
|
||||
|
||||
val keptCandidates = underlying.filterResult
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FilterResult")
|
||||
).result.asInstanceOf[Seq[Result]]
|
||||
|
||||
val decorations = underlying.candidateDecoratorResult
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "DecorationResult")
|
||||
).result.map(decoration => decoration.candidate -> decoration.presentation).toMap
|
||||
|
||||
val combinedFeatureMaps: Map[Result, FeatureMap] = previousResult.featureMaps.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "FeatureMaps"))
|
||||
|
||||
val filteredCandidates = keptCandidates.map { candidate =>
|
||||
val updatedMap = combinedFeatureMaps
|
||||
.get(candidate).getOrElse(FeatureMap.empty)
|
||||
FetchedCandidateWithFeatures(candidate, updatedMap)
|
||||
}
|
||||
|
||||
GroupResultsExecutorInput(
|
||||
candidates = filteredCandidates,
|
||||
decorations = decorations
|
||||
)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: IntermediateCandidatePipelineResult[Result],
|
||||
executorResult: GroupResultsExecutorResult
|
||||
): IntermediateCandidatePipelineResult[Result] =
|
||||
previousPipelineResult.copy(underlyingResult = previousPipelineResult.underlyingResult
|
||||
.copy(result = Some(executorResult.candidatesWithDetails)))
|
||||
}
|
||||
|
||||
val builtSteps = Seq(
|
||||
GatesStep,
|
||||
queryFeatureHydrationStep(
|
||||
config.queryFeatureHydration,
|
||||
CandidatePipelineConfig.fetchQueryFeaturesStep,
|
||||
(pipelineResult, executorResult) =>
|
||||
pipelineResult.copy(queryFeatures = Some(executorResult))
|
||||
),
|
||||
queryFeatureHydrationStep(
|
||||
config.queryFeatureHydrationPhase2,
|
||||
CandidatePipelineConfig.fetchQueryFeaturesPhase2Step,
|
||||
(pipelineResult, executorResult) =>
|
||||
pipelineResult.copy(
|
||||
queryFeaturesPhase2 = Some(executorResult),
|
||||
mergedAsyncQueryFeatures = Some(
|
||||
pipelineResult.queryFeatures
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(
|
||||
CandidatePipelineConfig.fetchQueryFeaturesPhase2Step,
|
||||
"QueryFeatures")
|
||||
).asyncFeatureMap ++ executorResult.asyncFeatureMap)
|
||||
)
|
||||
),
|
||||
asyncFeaturesStep(CandidatePipelineConfig.candidateSourceStep, context),
|
||||
CandidateSourceStep,
|
||||
asyncFeaturesStep(CandidatePipelineConfig.preFilterFeatureHydrationPhase1Step, context),
|
||||
PreFilterFeatureHydrationPhase1Step,
|
||||
asyncFeaturesStep(CandidatePipelineConfig.preFilterFeatureHydrationPhase2Step, context),
|
||||
PreFilterFeatureHydrationPhase2Step,
|
||||
asyncFeaturesStep(CandidatePipelineConfig.filtersStep, context),
|
||||
FiltersStep,
|
||||
asyncFeaturesStep(CandidatePipelineConfig.postFilterFeatureHydrationStep, context),
|
||||
PostFilterFeatureHydrationStep,
|
||||
asyncFeaturesStep(CandidatePipelineConfig.scorersStep, context),
|
||||
ScorersStep,
|
||||
asyncFeaturesStep(CandidatePipelineConfig.decoratorStep, context),
|
||||
DecorationStep,
|
||||
ResultStep
|
||||
)
|
||||
|
||||
/** The main execution logic for this Candidate Pipeline. */
|
||||
val finalArrow: Arrow[CandidatePipeline.Inputs[Query], CandidatePipelineResult] =
|
||||
buildCombinedArrowFromSteps(
|
||||
steps = builtSteps,
|
||||
context = context,
|
||||
initialEmptyResult =
|
||||
IntermediateCandidatePipelineResult.empty[Result](config.candidateSource.identifier),
|
||||
stepsInOrderFromConfig = CandidatePipelineConfig.stepsInOrder
|
||||
).map(_.underlyingResult)
|
||||
|
||||
val configFromBuilder = config
|
||||
new CandidatePipeline[Query] {
|
||||
override private[core] val config: BaseCandidatePipelineConfig[Query, _, _, _] =
|
||||
configFromBuilder
|
||||
override val arrow: Arrow[CandidatePipeline.Inputs[Query], CandidatePipelineResult] =
|
||||
finalArrow
|
||||
override val identifier: CandidatePipelineIdentifier = pipelineIdentifier
|
||||
override val alerts: Seq[Alert] = config.alerts
|
||||
override val children: Seq[Component] =
|
||||
allGates ++
|
||||
config.queryFeatureHydration ++
|
||||
Seq(queryTransformer, config.candidateSource, resultsTransformer) ++
|
||||
config.featuresFromCandidateSourceTransformers ++
|
||||
decorator.toSeq ++
|
||||
config.preFilterFeatureHydrationPhase1 ++
|
||||
config.filters ++
|
||||
config.postFilterFeatureHydration ++
|
||||
config.scorers
|
||||
}
|
||||
}
|
||||
|
||||
private case class CandidateWithFeaturesImpl(candidate: Result, features: FeatureMap)
|
||||
extends CandidateWithFeatures[Result]
|
||||
}
|
Binary file not shown.
@ -1,56 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.candidate
|
||||
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_decorator_executor.CandidateDecoratorExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_source_executor.CandidateSourceExecutor
|
||||
import com.twitter.product_mixer.core.service.filter_executor.FilterExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.group_results_executor.GroupResultsExecutor
|
||||
import com.twitter.product_mixer.core.service.query_feature_hydrator_executor.QueryFeatureHydratorExecutor
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
|
||||
@Singleton
|
||||
class CandidatePipelineBuilderFactory @Inject() (
|
||||
queryFeatureHydratorExecutor: QueryFeatureHydratorExecutor,
|
||||
asyncFeatureMapExecutor: AsyncFeatureMapExecutor,
|
||||
candidateDecoratorExecutor: CandidateDecoratorExecutor,
|
||||
candidateFeatureHydratorExecutor: CandidateFeatureHydratorExecutor,
|
||||
candidateSourceExecutor: CandidateSourceExecutor,
|
||||
groupResultsExecutor: GroupResultsExecutor,
|
||||
filterExecutor: FilterExecutor,
|
||||
gateExecutor: GateExecutor,
|
||||
statsReceiver: StatsReceiver) {
|
||||
def get[
|
||||
Query <: PipelineQuery,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result <: UniversalNoun[Any]
|
||||
]: CandidatePipelineBuilder[
|
||||
Query,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result
|
||||
] = {
|
||||
new CandidatePipelineBuilder[
|
||||
Query,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result
|
||||
](
|
||||
queryFeatureHydratorExecutor,
|
||||
asyncFeatureMapExecutor,
|
||||
candidateDecoratorExecutor,
|
||||
candidateFeatureHydratorExecutor,
|
||||
candidateSourceExecutor,
|
||||
groupResultsExecutor,
|
||||
filterExecutor,
|
||||
gateExecutor,
|
||||
statsReceiver
|
||||
)
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,264 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.candidate
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator
|
||||
import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator
|
||||
import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseQueryFeatureHydrator
|
||||
import com.twitter.product_mixer.core.functional_component.filter.Filter
|
||||
import com.twitter.product_mixer.core.functional_component.gate.BaseGate
|
||||
import com.twitter.product_mixer.core.functional_component.gate.Gate
|
||||
import com.twitter.product_mixer.core.functional_component.scorer.Scorer
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer
|
||||
import com.twitter.product_mixer.core.functional_component.transformer._
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfigCompanion
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.timelines.configapi.FSParam
|
||||
import com.twitter.timelines.configapi.decider.DeciderParam
|
||||
|
||||
sealed trait BaseCandidatePipelineConfig[
|
||||
-Query <: PipelineQuery,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result <: UniversalNoun[Any]]
|
||||
extends PipelineConfig {
|
||||
|
||||
val identifier: CandidatePipelineIdentifier
|
||||
|
||||
/**
|
||||
* A candidate pipeline can fetch query-level features for use within the candidate source. It's
|
||||
* generally recommended to set a hydrator in the parent recos or mixer pipeline if multiple
|
||||
* candidate pipelines share the same feature but if a specific query feature hydrator is used
|
||||
* by one pipeline and you don't want to block the others, you could explicitly set it here.
|
||||
* If a feature is hydrated both in the parent pipeline or here, this one takes priority.
|
||||
*/
|
||||
def queryFeatureHydration: Seq[BaseQueryFeatureHydrator[Query, _]] = Seq.empty
|
||||
|
||||
/**
|
||||
* For query-level features that are dependent on query-level features from [[queryFeatureHydration]]
|
||||
*/
|
||||
def queryFeatureHydrationPhase2: Seq[BaseQueryFeatureHydrator[Query, _]] = Seq.empty
|
||||
|
||||
/**
|
||||
* When these Params are defined, they will automatically be added as Gates in the pipeline
|
||||
* by the CandidatePipelineBuilder
|
||||
*
|
||||
* The enabled decider param can to be used to quickly disable a Candidate Pipeline via Decider
|
||||
*/
|
||||
val enabledDeciderParam: Option[DeciderParam[Boolean]] = None
|
||||
|
||||
/**
|
||||
* This supported client feature switch param can be used with a Feature Switch to control the
|
||||
* rollout of a new Candidate Pipeline from dogfood to experiment to production
|
||||
*/
|
||||
val supportedClientParam: Option[FSParam[Boolean]] = None
|
||||
|
||||
/** [[Gate]]s that are applied sequentially, the pipeline will only run if all the Gates are open */
|
||||
def gates: Seq[BaseGate[Query]] = Seq.empty
|
||||
|
||||
/**
|
||||
* A pair of transforms to adapt the underlying candidate source to the pipeline's query and result types
|
||||
* Complex use cases such as those that need access to features should construct their own transformer, but
|
||||
* for simple use cases, you can pass in an anonymous function.
|
||||
* @example
|
||||
* {{{ override val queryTransformer: CandidatePipelineQueryTransformer[Query, CandidateSourceQuery] = { query =>
|
||||
* query.toExampleThrift
|
||||
* }
|
||||
* }}}
|
||||
*/
|
||||
def queryTransformer: BaseCandidatePipelineQueryTransformer[
|
||||
Query,
|
||||
CandidateSourceQuery
|
||||
]
|
||||
|
||||
/** Source for Candidates for this Pipeline */
|
||||
def candidateSource: BaseCandidateSource[CandidateSourceQuery, CandidateSourceResult]
|
||||
|
||||
/**
|
||||
* [[CandidateFeatureTransformer]] allow you to define [[com.twitter.product_mixer.core.feature.Feature]] extraction logic from your [[CandidateSource]] results.
|
||||
* If your candidate sources return [[com.twitter.product_mixer.core.feature.Feature]]s alongside the candidate that might be useful later on,
|
||||
* add transformers for constructing feature maps.
|
||||
*
|
||||
* @note If multiple transformers extract the same feature, the last one takes priority and is kept.
|
||||
*/
|
||||
def featuresFromCandidateSourceTransformers: Seq[
|
||||
CandidateFeatureTransformer[CandidateSourceResult]
|
||||
] = Seq.empty
|
||||
|
||||
/**
|
||||
* a result Transformer may throw PipelineFailure for candidates that are malformed and
|
||||
* should be removed. This should be exceptional behavior, and not a replacement for adding a Filter.
|
||||
* Complex use cases such as those that need access to features should construct their own transformer, but
|
||||
* for simple use cases, you can pass in an anonymous function.
|
||||
* @example
|
||||
* {{{ override val queryTransformer: CandidatePipelineResultsTransformer[CandidateSourceResult, Result] = { sourceResult =>
|
||||
* ExampleCandidate(sourceResult.id)
|
||||
* }
|
||||
* }}}
|
||||
*
|
||||
*/
|
||||
val resultTransformer: CandidatePipelineResultsTransformer[CandidateSourceResult, Result]
|
||||
|
||||
/**
|
||||
* Before filters are run, you can fetch features for each candidate.
|
||||
*
|
||||
* Uses Stitch, so you're encouraged to use a working Stitch Adaptor to batch between candidates.
|
||||
*
|
||||
* The existing features (from the candidate source) are passed in as an input. You are not expected
|
||||
* to put them into the resulting feature map yourself - they will be merged for you by the platform.
|
||||
*
|
||||
* This API is likely to change when Product Mixer does managed feature hydration
|
||||
*/
|
||||
val preFilterFeatureHydrationPhase1: Seq[BaseCandidateFeatureHydrator[Query, Result, _]] =
|
||||
Seq.empty
|
||||
|
||||
/**
|
||||
* A second phase of feature hydration that can be run before filtering and after the first phase
|
||||
* of [[preFilterFeatureHydrationPhase1]]. You are not expected to put them into the resulting
|
||||
* feature map yourself - they will be merged for you by the platform.
|
||||
*/
|
||||
val preFilterFeatureHydrationPhase2: Seq[BaseCandidateFeatureHydrator[Query, Result, _]] =
|
||||
Seq.empty
|
||||
|
||||
/** A list of filters to apply. Filters will be applied in sequential order. */
|
||||
def filters: Seq[Filter[Query, Result]] = Seq.empty
|
||||
|
||||
/**
|
||||
* After filters are run, you can fetch features for each candidate.
|
||||
*
|
||||
* Uses Stitch, so you're encouraged to use a working Stitch Adaptor to batch between candidates.
|
||||
*
|
||||
* The existing features (from the candidate source) & pre-filtering are passed in as an input.
|
||||
* You are not expected to put them into the resulting feature map yourself -
|
||||
* they will be merged for you by the platform.
|
||||
*
|
||||
* This API is likely to change when Product Mixer does managed feature hydration
|
||||
*/
|
||||
val postFilterFeatureHydration: Seq[BaseCandidateFeatureHydrator[Query, Result, _]] = Seq.empty
|
||||
|
||||
/**
|
||||
* Decorators allow for adding Presentations to candidates. While the Presentation can contain any
|
||||
* arbitrary data, Decorators are often used to add a UrtItemPresentation for URT item support, or
|
||||
* a UrtModulePresentation for grouping the candidates in a URT module.
|
||||
*/
|
||||
val decorator: Option[CandidateDecorator[Query, Result]] = None
|
||||
|
||||
/**
|
||||
* A candidate pipeline can define a partial function to rescue failures here. They will be treated as failures
|
||||
* from a monitoring standpoint, and cancellation exceptions will always be propagated (they cannot be caught here).
|
||||
*/
|
||||
def failureClassifier: PartialFunction[Throwable, PipelineFailure] = PartialFunction.empty
|
||||
|
||||
/**
|
||||
* Scorers for candidates. Scorers are executed in parallel. Order does not matter.
|
||||
*/
|
||||
def scorers: Seq[Scorer[Query, Result]] = Seq.empty
|
||||
|
||||
/**
|
||||
* Alerts can be used to indicate the pipeline's service level objectives. Alerts and
|
||||
* dashboards will be automatically created based on this information.
|
||||
*/
|
||||
val alerts: Seq[Alert] = Seq.empty
|
||||
|
||||
/**
|
||||
* This method is used by the product mixer framework to build the pipeline.
|
||||
*/
|
||||
private[core] final def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
factory: CandidatePipelineBuilderFactory
|
||||
): CandidatePipeline[Query] = {
|
||||
factory.get.build(parentComponentIdentifierStack, this)
|
||||
}
|
||||
}
|
||||
|
||||
trait CandidatePipelineConfig[
|
||||
-Query <: PipelineQuery,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result <: UniversalNoun[Any]]
|
||||
extends BaseCandidatePipelineConfig[
|
||||
Query,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result
|
||||
] {
|
||||
override val gates: Seq[Gate[Query]] = Seq.empty
|
||||
|
||||
override val queryTransformer: CandidatePipelineQueryTransformer[
|
||||
Query,
|
||||
CandidateSourceQuery
|
||||
]
|
||||
}
|
||||
|
||||
trait DependentCandidatePipelineConfig[
|
||||
-Query <: PipelineQuery,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result <: UniversalNoun[Any]]
|
||||
extends BaseCandidatePipelineConfig[
|
||||
Query,
|
||||
CandidateSourceQuery,
|
||||
CandidateSourceResult,
|
||||
Result
|
||||
]
|
||||
|
||||
/**
|
||||
* Contains [[PipelineStepIdentifier]]s for the Steps that are available for all [[BaseCandidatePipelineConfig]]s
|
||||
*/
|
||||
object CandidatePipelineConfig extends PipelineConfigCompanion {
|
||||
val gatesStep: PipelineStepIdentifier = PipelineStepIdentifier("Gates")
|
||||
val fetchQueryFeaturesStep: PipelineStepIdentifier = PipelineStepIdentifier("FetchQueryFeatures")
|
||||
val fetchQueryFeaturesPhase2Step: PipelineStepIdentifier = PipelineStepIdentifier(
|
||||
"FetchQueryFeaturesPhase2")
|
||||
val candidateSourceStep: PipelineStepIdentifier = PipelineStepIdentifier("CandidateSource")
|
||||
val preFilterFeatureHydrationPhase1Step: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("PreFilterFeatureHydration")
|
||||
val preFilterFeatureHydrationPhase2Step: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("PreFilterFeatureHydrationPhase2")
|
||||
val filtersStep: PipelineStepIdentifier = PipelineStepIdentifier("Filters")
|
||||
val postFilterFeatureHydrationStep: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("PostFilterFeatureHydration")
|
||||
val scorersStep: PipelineStepIdentifier = PipelineStepIdentifier("Scorer")
|
||||
val decoratorStep: PipelineStepIdentifier = PipelineStepIdentifier("Decorator")
|
||||
val resultStep: PipelineStepIdentifier = PipelineStepIdentifier("Result")
|
||||
|
||||
/** All the steps which are executed by a [[CandidatePipeline]] in the order in which they are run */
|
||||
override val stepsInOrder: Seq[PipelineStepIdentifier] = Seq(
|
||||
gatesStep,
|
||||
fetchQueryFeaturesStep,
|
||||
fetchQueryFeaturesPhase2Step,
|
||||
asyncFeaturesStep(candidateSourceStep),
|
||||
candidateSourceStep,
|
||||
asyncFeaturesStep(preFilterFeatureHydrationPhase1Step),
|
||||
preFilterFeatureHydrationPhase1Step,
|
||||
asyncFeaturesStep(preFilterFeatureHydrationPhase2Step),
|
||||
preFilterFeatureHydrationPhase2Step,
|
||||
asyncFeaturesStep(filtersStep),
|
||||
filtersStep,
|
||||
asyncFeaturesStep(postFilterFeatureHydrationStep),
|
||||
postFilterFeatureHydrationStep,
|
||||
asyncFeaturesStep(scorersStep),
|
||||
scorersStep,
|
||||
asyncFeaturesStep(decoratorStep),
|
||||
decoratorStep,
|
||||
resultStep
|
||||
)
|
||||
|
||||
override val stepsAsyncFeatureHydrationCanBeCompletedBy: Set[PipelineStepIdentifier] = Set(
|
||||
candidateSourceStep,
|
||||
preFilterFeatureHydrationPhase1Step,
|
||||
preFilterFeatureHydrationPhase2Step,
|
||||
filtersStep,
|
||||
postFilterFeatureHydrationStep,
|
||||
scorersStep,
|
||||
decoratorStep
|
||||
)
|
||||
}
|
Binary file not shown.
@ -1,93 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.candidate
|
||||
|
||||
import com.twitter.product_mixer.core.feature.featuremap.FeatureMap
|
||||
import com.twitter.product_mixer.core.feature.featuremap.asyncfeaturemap.AsyncFeatureMap
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineResult
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutorResults
|
||||
import com.twitter.product_mixer.core.service.candidate_decorator_executor.CandidateDecoratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.candidate_source_executor.CandidateSourceExecutorResult
|
||||
import com.twitter.product_mixer.core.service.filter_executor.FilterExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.query_feature_hydrator_executor.QueryFeatureHydratorExecutor
|
||||
|
||||
case class CandidatePipelineResult(
|
||||
candidateSourceIdentifier: CandidateSourceIdentifier,
|
||||
gateResult: Option[GateExecutorResult],
|
||||
queryFeatures: Option[QueryFeatureHydratorExecutor.Result],
|
||||
queryFeaturesPhase2: Option[QueryFeatureHydratorExecutor.Result],
|
||||
mergedAsyncQueryFeatures: Option[AsyncFeatureMap],
|
||||
candidateSourceResult: Option[CandidateSourceExecutorResult[UniversalNoun[Any]]],
|
||||
preFilterHydrationResult: Option[CandidateFeatureHydratorExecutorResult[UniversalNoun[Any]]],
|
||||
preFilterHydrationResultPhase2: Option[
|
||||
CandidateFeatureHydratorExecutorResult[UniversalNoun[Any]]
|
||||
],
|
||||
filterResult: Option[FilterExecutorResult[UniversalNoun[Any]]],
|
||||
postFilterHydrationResult: Option[CandidateFeatureHydratorExecutorResult[UniversalNoun[Any]]],
|
||||
candidateDecoratorResult: Option[CandidateDecoratorExecutorResult],
|
||||
scorersResult: Option[CandidateFeatureHydratorExecutorResult[UniversalNoun[Any]]],
|
||||
asyncFeatureHydrationResults: Option[AsyncFeatureMapExecutorResults],
|
||||
failure: Option[PipelineFailure],
|
||||
result: Option[Seq[CandidateWithDetails]])
|
||||
extends PipelineResult[Seq[CandidateWithDetails]] {
|
||||
|
||||
override def withFailure(failure: PipelineFailure): CandidatePipelineResult =
|
||||
copy(failure = Some(failure))
|
||||
|
||||
override def withResult(
|
||||
result: Seq[CandidateWithDetails]
|
||||
): CandidatePipelineResult = copy(result = Some(result))
|
||||
|
||||
override val resultSize: Int = result.map(PipelineResult.resultSize).getOrElse(0)
|
||||
}
|
||||
|
||||
private[candidate] object IntermediateCandidatePipelineResult {
|
||||
def empty[Candidate <: UniversalNoun[Any]](
|
||||
candidateSourceIdentifier: CandidateSourceIdentifier
|
||||
): IntermediateCandidatePipelineResult[Candidate] = {
|
||||
IntermediateCandidatePipelineResult(
|
||||
CandidatePipelineResult(
|
||||
candidateSourceIdentifier = candidateSourceIdentifier,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None
|
||||
),
|
||||
None
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
private[candidate] case class IntermediateCandidatePipelineResult[Candidate <: UniversalNoun[Any]](
|
||||
underlyingResult: CandidatePipelineResult,
|
||||
featureMaps: Option[Map[Candidate, FeatureMap]])
|
||||
extends PipelineResult[Seq[CandidateWithDetails]] {
|
||||
override val failure: Option[PipelineFailure] = underlyingResult.failure
|
||||
override val result: Option[Seq[CandidateWithDetails]] = underlyingResult.result
|
||||
|
||||
override def withFailure(
|
||||
failure: PipelineFailure
|
||||
): IntermediateCandidatePipelineResult[Candidate] =
|
||||
copy(underlyingResult = underlyingResult.withFailure(failure))
|
||||
|
||||
override def withResult(
|
||||
result: Seq[CandidateWithDetails]
|
||||
): IntermediateCandidatePipelineResult[Candidate] =
|
||||
copy(underlyingResult = underlyingResult.withResult(result))
|
||||
|
||||
override def resultSize(): Int = underlyingResult.resultSize
|
||||
}
|
Binary file not shown.
@ -1,47 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.candidate
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.candidate_source.PassthroughCandidateSource
|
||||
import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateExtractor
|
||||
import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
|
||||
object PassthroughCandidatePipelineConfig {
|
||||
|
||||
/**
|
||||
* Build a [[PassthroughCandidatePipelineConfig]] with a [[PassthroughCandidateSource]] built from
|
||||
* a [[CandidateExtractor]]
|
||||
*/
|
||||
def apply[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]](
|
||||
identifier: CandidatePipelineIdentifier,
|
||||
extractor: CandidateExtractor[Query, Candidate],
|
||||
decorator: Option[CandidateDecorator[Query, Candidate]] = None
|
||||
): PassthroughCandidatePipelineConfig[Query, Candidate] = {
|
||||
|
||||
// Renaming variables to keep the interface clean, but avoid naming collisions when creating
|
||||
// the anonymous class.
|
||||
val _identifier = identifier
|
||||
val _extractor = extractor
|
||||
val _decorator = decorator
|
||||
|
||||
new PassthroughCandidatePipelineConfig[Query, Candidate] {
|
||||
override val identifier = _identifier
|
||||
override val candidateSource =
|
||||
PassthroughCandidateSource(CandidateSourceIdentifier(_identifier.name), _extractor)
|
||||
override val decorator = _decorator
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
trait PassthroughCandidatePipelineConfig[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]]
|
||||
extends CandidatePipelineConfig[Query, Query, Candidate, Candidate] {
|
||||
|
||||
override val queryTransformer: CandidatePipelineQueryTransformer[Query, Query] = identity
|
||||
|
||||
override val resultTransformer: CandidatePipelineResultsTransformer[Candidate, Candidate] =
|
||||
identity
|
||||
}
|
Binary file not shown.
@ -1,51 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.candidate
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource
|
||||
import com.twitter.product_mixer.core.functional_component.candidate_source.StaticCandidateSource
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer
|
||||
import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator
|
||||
|
||||
object StaticCandidatePipelineConfig {
|
||||
|
||||
/**
|
||||
* Build a [[StaticCandidatePipelineConfig]] with a [[CandidateSource]] that returns the [[candidate]]
|
||||
*/
|
||||
def apply[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]](
|
||||
identifier: CandidatePipelineIdentifier,
|
||||
candidate: Candidate,
|
||||
decorator: Option[CandidateDecorator[Query, Candidate]] = None
|
||||
): StaticCandidatePipelineConfig[Query, Candidate] = {
|
||||
|
||||
// Renaming variables to keep the interface clean, but avoid naming collisions when creating
|
||||
// the anonymous class.
|
||||
val _identifier = identifier
|
||||
val _candidate = candidate
|
||||
val _decorator = decorator
|
||||
|
||||
new StaticCandidatePipelineConfig[Query, Candidate] {
|
||||
override val identifier = _identifier
|
||||
override val candidate = _candidate
|
||||
override val decorator = _decorator
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
trait StaticCandidatePipelineConfig[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]]
|
||||
extends CandidatePipelineConfig[Query, Unit, Unit, Candidate] {
|
||||
|
||||
val candidate: Candidate
|
||||
|
||||
override def candidateSource: CandidateSource[Unit, Unit] = StaticCandidateSource[Unit](
|
||||
identifier = CandidateSourceIdentifier(identifier.name),
|
||||
result = Seq(()))
|
||||
|
||||
override val queryTransformer: CandidatePipelineQueryTransformer[Query, Unit] = _ => Unit
|
||||
|
||||
override val resultTransformer: CandidatePipelineResultsTransformer[Unit, Candidate] = _ =>
|
||||
candidate
|
||||
}
|
@ -1,62 +0,0 @@
|
||||
scala_library(
|
||||
sources = ["*.scala"],
|
||||
compiler_option_sets = ["fatal_warnings"],
|
||||
platform = "java8",
|
||||
strict_deps = True,
|
||||
tags = ["bazel-compatible"],
|
||||
dependencies = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featurestorev1",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/premarshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/selector",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/side_effect",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/async_feature_map_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_pipeline_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/domain_marshaller_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_result_side_effect_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/quality_factor_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/selector_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/transport_marshaller_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util",
|
||||
"stitch/stitch-core",
|
||||
],
|
||||
exports = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/premarshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/selector",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/side_effect",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_pipeline_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/domain_marshaller_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_result_side_effect_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/quality_factor_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/selector_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/transport_marshaller_executor",
|
||||
"stitch/stitch-core",
|
||||
],
|
||||
)
|
Binary file not shown.
Binary file not shown.
@ -1,24 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.mixer
|
||||
|
||||
import com.twitter.product_mixer.core.model.common.identifier.MixerPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.pipeline.Pipeline
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.stitch.Arrow
|
||||
|
||||
/**
|
||||
* A Mixer Pipeline
|
||||
*
|
||||
* This is an abstract class, as we only construct these via the [[MixerPipelineBuilder]].
|
||||
*
|
||||
* A [[MixerPipeline]] is capable of processing requests (queries) and returning responses (results)
|
||||
* in the correct format to directly send to users.
|
||||
*
|
||||
* @tparam Query the domain model for the query or request
|
||||
* @tparam Result the final marshalled result type
|
||||
*/
|
||||
abstract class MixerPipeline[Query <: PipelineQuery, Result] private[mixer]
|
||||
extends Pipeline[Query, Result] {
|
||||
override private[core] val config: MixerPipelineConfig[Query, _, Result]
|
||||
override val arrow: Arrow[Query, MixerPipelineResult[Result]]
|
||||
override val identifier: MixerPipelineIdentifier
|
||||
}
|
Binary file not shown.
@ -1,582 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.mixer
|
||||
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.product_mixer.core.feature.featuremap.FeatureMap
|
||||
import com.twitter.product_mixer.core.feature.featuremap.asyncfeaturemap.AsyncFeatureMap
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator
|
||||
import com.twitter.product_mixer.core.functional_component.gate.Gate
|
||||
import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller
|
||||
import com.twitter.product_mixer.core.functional_component.selector.Selector
|
||||
import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect
|
||||
import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller
|
||||
import com.twitter.product_mixer.core.model.common.Component
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.MixerPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.model.marshalling.HasMarshalling
|
||||
import com.twitter.product_mixer.core.pipeline.FailOpenPolicy
|
||||
import com.twitter.product_mixer.core.pipeline.InvalidStepStateException
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineBuilder
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipeline
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineBuilderFactory
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailureClassifier
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.ProductDisabled
|
||||
import com.twitter.product_mixer.core.quality_factor.HasQualityFactorStatus
|
||||
import com.twitter.product_mixer.core.quality_factor.QualityFactorObserver
|
||||
import com.twitter.product_mixer.core.quality_factor.QualityFactorStatus
|
||||
import com.twitter.product_mixer.core.service.Executor
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutor
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutorResults
|
||||
import com.twitter.product_mixer.core.service.candidate_pipeline_executor.CandidatePipelineExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_pipeline_executor.CandidatePipelineExecutorResult
|
||||
import com.twitter.product_mixer.core.service.domain_marshaller_executor.DomainMarshallerExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.StoppedGateException
|
||||
import com.twitter.product_mixer.core.service.pipeline_result_side_effect_executor.PipelineResultSideEffectExecutor
|
||||
import com.twitter.product_mixer.core.service.quality_factor_executor.QualityFactorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.query_feature_hydrator_executor.QueryFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutor
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.transport_marshaller_executor.TransportMarshallerExecutor
|
||||
import com.twitter.stitch.Arrow
|
||||
import com.twitter.util.logging.Logging
|
||||
|
||||
/**
|
||||
* MixerPipelineBuilder builds [[MixerPipeline]]s from [[MixerPipelineConfig]]s.
|
||||
*
|
||||
* You should inject a [[MixerPipelineBuilderFactory]] and call `.get` to build these.
|
||||
*
|
||||
* @see [[MixerPipelineConfig]] for the description of the type parameters
|
||||
*/
|
||||
class MixerPipelineBuilder[Query <: PipelineQuery, DomainResultType <: HasMarshalling, Result](
|
||||
candidatePipelineExecutor: CandidatePipelineExecutor,
|
||||
gateExecutor: GateExecutor,
|
||||
selectorExecutor: SelectorExecutor,
|
||||
queryFeatureHydratorExecutor: QueryFeatureHydratorExecutor,
|
||||
asyncFeatureMapExecutor: AsyncFeatureMapExecutor,
|
||||
domainMarshallerExecutor: DomainMarshallerExecutor,
|
||||
transportMarshallerExecutor: TransportMarshallerExecutor,
|
||||
pipelineResultSideEffectExecutor: PipelineResultSideEffectExecutor,
|
||||
candidatePipelineBuilderFactory: CandidatePipelineBuilderFactory,
|
||||
override val statsReceiver: StatsReceiver)
|
||||
extends PipelineBuilder[Query]
|
||||
with Logging {
|
||||
|
||||
override type UnderlyingResultType = Result
|
||||
override type PipelineResultType = MixerPipelineResult[Result]
|
||||
|
||||
def qualityFactorStep(
|
||||
qualityFactorStatus: QualityFactorStatus
|
||||
): Step[Query, QualityFactorExecutorResult] =
|
||||
new Step[Query, QualityFactorExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = MixerPipelineConfig.qualityFactorStep
|
||||
|
||||
override def executorArrow: Arrow[Query, QualityFactorExecutorResult] =
|
||||
Arrow
|
||||
.map[Query, QualityFactorExecutorResult] { _ =>
|
||||
QualityFactorExecutorResult(
|
||||
pipelineQualityFactors =
|
||||
qualityFactorStatus.qualityFactorByPipeline.mapValues(_.currentValue)
|
||||
)
|
||||
}
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): Query = query
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: QualityFactorExecutorResult
|
||||
): MixerPipelineResult[Result] =
|
||||
previousPipelineResult.copy(qualityFactorResult = Some(executorResult))
|
||||
|
||||
override def queryUpdater(
|
||||
query: Query,
|
||||
executorResult: QualityFactorExecutorResult
|
||||
): Query = {
|
||||
query match {
|
||||
case queryWithQualityFactor: HasQualityFactorStatus =>
|
||||
queryWithQualityFactor
|
||||
.withQualityFactorStatus(
|
||||
queryWithQualityFactor.qualityFactorStatus.getOrElse(QualityFactorStatus.empty) ++
|
||||
qualityFactorStatus
|
||||
).asInstanceOf[Query]
|
||||
case _ =>
|
||||
query
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
def gatesStep(
|
||||
gates: Seq[Gate[Query]],
|
||||
context: Executor.Context
|
||||
): Step[Query, GateExecutorResult] = new Step[Query, GateExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = MixerPipelineConfig.gatesStep
|
||||
|
||||
override def executorArrow: Arrow[Query, GateExecutorResult] =
|
||||
gateExecutor.arrow(gates, context)
|
||||
|
||||
override def inputAdaptor(query: Query, previousResult: MixerPipelineResult[Result]): Query =
|
||||
query
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: GateExecutorResult
|
||||
): MixerPipelineResult[Result] =
|
||||
previousPipelineResult.copy(gateResult = Some(executorResult))
|
||||
}
|
||||
|
||||
def fetchQueryFeaturesStep(
|
||||
queryFeatureHydrators: Seq[QueryFeatureHydrator[Query]],
|
||||
stepIdentifier: PipelineStepIdentifier,
|
||||
updater: ResultUpdater[MixerPipelineResult[Result], QueryFeatureHydratorExecutor.Result],
|
||||
context: Executor.Context
|
||||
): Step[Query, QueryFeatureHydratorExecutor.Result] =
|
||||
new Step[Query, QueryFeatureHydratorExecutor.Result] {
|
||||
override def identifier: PipelineStepIdentifier = stepIdentifier
|
||||
|
||||
override def executorArrow: Arrow[Query, QueryFeatureHydratorExecutor.Result] =
|
||||
queryFeatureHydratorExecutor.arrow(
|
||||
queryFeatureHydrators,
|
||||
MixerPipelineConfig.stepsAsyncFeatureHydrationCanBeCompletedBy,
|
||||
context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): Query = query
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: QueryFeatureHydratorExecutor.Result
|
||||
): MixerPipelineResult[Result] =
|
||||
updater(previousPipelineResult, executorResult)
|
||||
|
||||
override def queryUpdater(
|
||||
query: Query,
|
||||
executorResult: QueryFeatureHydratorExecutor.Result
|
||||
): Query =
|
||||
query
|
||||
.withFeatureMap(
|
||||
query.features
|
||||
.getOrElse(FeatureMap.empty) ++ executorResult.featureMap).asInstanceOf[Query]
|
||||
}
|
||||
|
||||
def asyncFeaturesStep(
|
||||
stepToHydrateFor: PipelineStepIdentifier,
|
||||
context: Executor.Context
|
||||
): Step[AsyncFeatureMap, AsyncFeatureMapExecutorResults] =
|
||||
new Step[AsyncFeatureMap, AsyncFeatureMapExecutorResults] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
MixerPipelineConfig.asyncFeaturesStep(stepToHydrateFor)
|
||||
|
||||
override def executorArrow: Arrow[AsyncFeatureMap, AsyncFeatureMapExecutorResults] =
|
||||
asyncFeatureMapExecutor.arrow(
|
||||
stepToHydrateFor,
|
||||
identifier,
|
||||
context
|
||||
)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): AsyncFeatureMap =
|
||||
previousResult.mergedAsyncQueryFeatures
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "MergedAsyncQueryFeatures")
|
||||
)
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: AsyncFeatureMapExecutorResults
|
||||
): MixerPipelineResult[Result] = previousPipelineResult.copy(
|
||||
asyncFeatureHydrationResults = previousPipelineResult.asyncFeatureHydrationResults match {
|
||||
case Some(existingResults) => Some(existingResults ++ executorResult)
|
||||
case None => Some(executorResult)
|
||||
})
|
||||
|
||||
override def queryUpdater(
|
||||
query: Query,
|
||||
executorResult: AsyncFeatureMapExecutorResults
|
||||
): Query =
|
||||
if (executorResult.featureMapsByStep
|
||||
.getOrElse(stepToHydrateFor, FeatureMap.empty).isEmpty) {
|
||||
query
|
||||
} else {
|
||||
query
|
||||
.withFeatureMap(
|
||||
query.features
|
||||
.getOrElse(FeatureMap.empty) ++ executorResult.featureMapsByStep(
|
||||
stepToHydrateFor)).asInstanceOf[Query]
|
||||
}
|
||||
}
|
||||
|
||||
def candidatePipelinesStep(
|
||||
candidatePipelines: Seq[CandidatePipeline[Query]],
|
||||
defaultFailOpenPolicy: FailOpenPolicy,
|
||||
failOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy],
|
||||
qualityFactorObserverByPipeline: Map[ComponentIdentifier, QualityFactorObserver],
|
||||
context: Executor.Context
|
||||
): Step[CandidatePipeline.Inputs[Query], CandidatePipelineExecutorResult] =
|
||||
new Step[CandidatePipeline.Inputs[Query], CandidatePipelineExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = MixerPipelineConfig.candidatePipelinesStep
|
||||
|
||||
override def executorArrow: Arrow[CandidatePipeline.Inputs[
|
||||
Query
|
||||
], CandidatePipelineExecutorResult] =
|
||||
candidatePipelineExecutor
|
||||
.arrow(
|
||||
candidatePipelines,
|
||||
defaultFailOpenPolicy,
|
||||
failOpenPolicies,
|
||||
qualityFactorObserverByPipeline,
|
||||
context
|
||||
)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): CandidatePipeline.Inputs[Query] = CandidatePipeline.Inputs[Query](query, Seq.empty)
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: CandidatePipelineExecutorResult
|
||||
): MixerPipelineResult[Result] =
|
||||
previousPipelineResult.copy(candidatePipelineResults = Some(executorResult))
|
||||
|
||||
override def queryUpdater(
|
||||
query: Query,
|
||||
executorResult: CandidatePipelineExecutorResult
|
||||
): Query = {
|
||||
val updatedFeatureMap = query.features
|
||||
.getOrElse(FeatureMap.empty) ++ executorResult.queryFeatureMap
|
||||
query
|
||||
.withFeatureMap(updatedFeatureMap).asInstanceOf[Query]
|
||||
}
|
||||
}
|
||||
|
||||
def dependentCandidatePipelinesStep(
|
||||
candidatePipelines: Seq[CandidatePipeline[Query]],
|
||||
defaultFailOpenPolicy: FailOpenPolicy,
|
||||
failOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy],
|
||||
qualityFactorObserverByPipeline: Map[ComponentIdentifier, QualityFactorObserver],
|
||||
context: Executor.Context
|
||||
): Step[CandidatePipeline.Inputs[Query], CandidatePipelineExecutorResult] =
|
||||
new Step[CandidatePipeline.Inputs[Query], CandidatePipelineExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
MixerPipelineConfig.dependentCandidatePipelinesStep
|
||||
|
||||
override def executorArrow: Arrow[CandidatePipeline.Inputs[
|
||||
Query
|
||||
], CandidatePipelineExecutorResult] =
|
||||
candidatePipelineExecutor
|
||||
.arrow(
|
||||
candidatePipelines,
|
||||
defaultFailOpenPolicy,
|
||||
failOpenPolicies,
|
||||
qualityFactorObserverByPipeline,
|
||||
context
|
||||
)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): CandidatePipeline.Inputs[Query] = {
|
||||
val previousCandidates = previousResult.candidatePipelineResults
|
||||
.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "Candidates")
|
||||
}.candidatePipelineResults.flatMap(_.result.getOrElse(Seq.empty))
|
||||
CandidatePipeline.Inputs[Query](query, previousCandidates)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: CandidatePipelineExecutorResult
|
||||
): MixerPipelineResult[Result] =
|
||||
previousPipelineResult.copy(dependentCandidatePipelineResults = Some(executorResult))
|
||||
|
||||
override def queryUpdater(
|
||||
query: Query,
|
||||
executorResult: CandidatePipelineExecutorResult
|
||||
): Query = {
|
||||
val updatedFeatureMap = query.features
|
||||
.getOrElse(FeatureMap.empty) ++ executorResult.queryFeatureMap
|
||||
query
|
||||
.withFeatureMap(updatedFeatureMap).asInstanceOf[Query]
|
||||
}
|
||||
}
|
||||
|
||||
def resultSelectorsStep(
|
||||
selectors: Seq[Selector[Query]],
|
||||
context: Executor.Context
|
||||
): Step[SelectorExecutor.Inputs[Query], SelectorExecutorResult] =
|
||||
new Step[SelectorExecutor.Inputs[Query], SelectorExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = MixerPipelineConfig.resultSelectorsStep
|
||||
|
||||
override def executorArrow: Arrow[SelectorExecutor.Inputs[Query], SelectorExecutorResult] =
|
||||
selectorExecutor.arrow(selectors, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): SelectorExecutor.Inputs[Query] = {
|
||||
val candidates = previousResult.candidatePipelineResults
|
||||
.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "Candidates")
|
||||
}.candidatePipelineResults.flatMap(_.result.getOrElse(Seq.empty))
|
||||
|
||||
val dependentCandidates =
|
||||
previousResult.dependentCandidatePipelineResults
|
||||
.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "DependentCandidates")
|
||||
}.candidatePipelineResults.flatMap(_.result.getOrElse(Seq.empty))
|
||||
|
||||
SelectorExecutor.Inputs(
|
||||
query = query,
|
||||
candidatesWithDetails = candidates ++ dependentCandidates
|
||||
)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: SelectorExecutorResult
|
||||
): MixerPipelineResult[Result] =
|
||||
previousPipelineResult.copy(resultSelectorResults = Some(executorResult))
|
||||
}
|
||||
|
||||
def domainMarshallingStep(
|
||||
domainMarshaller: DomainMarshaller[Query, DomainResultType],
|
||||
context: Executor.Context
|
||||
): Step[DomainMarshallerExecutor.Inputs[Query], DomainMarshallerExecutor.Result[
|
||||
DomainResultType
|
||||
]] =
|
||||
new Step[DomainMarshallerExecutor.Inputs[Query], DomainMarshallerExecutor.Result[
|
||||
DomainResultType
|
||||
]] {
|
||||
override def identifier: PipelineStepIdentifier = MixerPipelineConfig.domainMarshallerStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
DomainMarshallerExecutor.Inputs[Query],
|
||||
DomainMarshallerExecutor.Result[DomainResultType]
|
||||
] =
|
||||
domainMarshallerExecutor.arrow(domainMarshaller, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): DomainMarshallerExecutor.Inputs[Query] = {
|
||||
val selectorResults = previousResult.resultSelectorResults.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "SelectorResults")
|
||||
}
|
||||
|
||||
DomainMarshallerExecutor.Inputs(
|
||||
query = query,
|
||||
candidatesWithDetails = selectorResults.selectedCandidates
|
||||
)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: DomainMarshallerExecutor.Result[DomainResultType]
|
||||
): MixerPipelineResult[Result] = previousPipelineResult.copy(
|
||||
domainMarshallerResults = Some(executorResult)
|
||||
)
|
||||
}
|
||||
|
||||
def resultSideEffectsStep(
|
||||
sideEffects: Seq[PipelineResultSideEffect[Query, DomainResultType]],
|
||||
context: Executor.Context
|
||||
): Step[
|
||||
PipelineResultSideEffect.Inputs[Query, DomainResultType],
|
||||
PipelineResultSideEffectExecutor.Result
|
||||
] = new Step[
|
||||
PipelineResultSideEffect.Inputs[Query, DomainResultType],
|
||||
PipelineResultSideEffectExecutor.Result
|
||||
] {
|
||||
override def identifier: PipelineStepIdentifier = MixerPipelineConfig.resultSideEffectsStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
PipelineResultSideEffect.Inputs[Query, DomainResultType],
|
||||
PipelineResultSideEffectExecutor.Result
|
||||
] = pipelineResultSideEffectExecutor.arrow(sideEffects, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): PipelineResultSideEffect.Inputs[Query, DomainResultType] = {
|
||||
|
||||
val selectorResults = previousResult.resultSelectorResults.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "SelectorResults")
|
||||
}
|
||||
|
||||
val domainMarshallerResults = previousResult.domainMarshallerResults.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "DomainMarshallerResults")
|
||||
}
|
||||
|
||||
PipelineResultSideEffect.Inputs[Query, DomainResultType](
|
||||
query = query,
|
||||
selectedCandidates = selectorResults.selectedCandidates,
|
||||
remainingCandidates = selectorResults.remainingCandidates,
|
||||
droppedCandidates = selectorResults.droppedCandidates,
|
||||
response = domainMarshallerResults.result.asInstanceOf[DomainResultType]
|
||||
)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: PipelineResultSideEffectExecutor.Result
|
||||
): MixerPipelineResult[Result] =
|
||||
previousPipelineResult.copy(resultSideEffectResults = Some(executorResult))
|
||||
}
|
||||
|
||||
def transportMarshallingStep(
|
||||
transportMarshaller: TransportMarshaller[DomainResultType, Result],
|
||||
context: Executor.Context
|
||||
): Step[
|
||||
TransportMarshallerExecutor.Inputs[DomainResultType],
|
||||
TransportMarshallerExecutor.Result[Result]
|
||||
] = new Step[TransportMarshallerExecutor.Inputs[
|
||||
DomainResultType
|
||||
], TransportMarshallerExecutor.Result[Result]] {
|
||||
override def identifier: PipelineStepIdentifier = MixerPipelineConfig.transportMarshallerStep
|
||||
|
||||
override def executorArrow: Arrow[TransportMarshallerExecutor.Inputs[
|
||||
DomainResultType
|
||||
], TransportMarshallerExecutor.Result[Result]] =
|
||||
transportMarshallerExecutor.arrow(transportMarshaller, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Query,
|
||||
previousResult: MixerPipelineResult[Result]
|
||||
): TransportMarshallerExecutor.Inputs[DomainResultType] = {
|
||||
val domainMarshallingResults = previousResult.domainMarshallerResults.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "DomainMarshallerResults")
|
||||
}
|
||||
|
||||
// Since the PipelineResult just uses HasMarshalling
|
||||
val domainResult = domainMarshallingResults.result.asInstanceOf[DomainResultType]
|
||||
|
||||
TransportMarshallerExecutor.Inputs(domainResult)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: MixerPipelineResult[Result],
|
||||
executorResult: TransportMarshallerExecutor.Result[Result]
|
||||
): MixerPipelineResult[Result] = previousPipelineResult.copy(
|
||||
transportMarshallerResults = Some(executorResult),
|
||||
result = Some(executorResult.result)
|
||||
)
|
||||
}
|
||||
|
||||
def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
config: MixerPipelineConfig[Query, DomainResultType, Result]
|
||||
): MixerPipeline[Query, Result] = {
|
||||
|
||||
val pipelineIdentifier = config.identifier
|
||||
|
||||
val context = Executor.Context(
|
||||
PipelineFailureClassifier(
|
||||
config.failureClassifier.orElse(StoppedGateException.classifier(ProductDisabled))),
|
||||
parentComponentIdentifierStack.push(pipelineIdentifier)
|
||||
)
|
||||
|
||||
val qualityFactorStatus: QualityFactorStatus =
|
||||
QualityFactorStatus.build(config.qualityFactorConfigs)
|
||||
|
||||
val qualityFactorObserverByPipeline =
|
||||
qualityFactorStatus.qualityFactorByPipeline.mapValues { qualityFactor =>
|
||||
qualityFactor.buildObserver()
|
||||
}
|
||||
|
||||
buildGaugesForQualityFactor(pipelineIdentifier, qualityFactorStatus, statsReceiver)
|
||||
|
||||
val candidatePipelines: Seq[CandidatePipeline[Query]] = config.candidatePipelines.map {
|
||||
pipelineConfig: CandidatePipelineConfig[Query, _, _, _] =>
|
||||
pipelineConfig.build(context.componentStack, candidatePipelineBuilderFactory)
|
||||
}
|
||||
|
||||
val dependentCandidatePipelines: Seq[CandidatePipeline[Query]] =
|
||||
config.dependentCandidatePipelines.map {
|
||||
pipelineConfig: DependentCandidatePipelineConfig[Query, _, _, _] =>
|
||||
pipelineConfig.build(context.componentStack, candidatePipelineBuilderFactory)
|
||||
}
|
||||
|
||||
val builtSteps = Seq(
|
||||
qualityFactorStep(qualityFactorStatus),
|
||||
gatesStep(config.gates, context),
|
||||
fetchQueryFeaturesStep(
|
||||
config.fetchQueryFeatures,
|
||||
MixerPipelineConfig.fetchQueryFeaturesStep,
|
||||
(previousPipelineResult, executorResult) =>
|
||||
previousPipelineResult.copy(queryFeatures = Some(executorResult)),
|
||||
context
|
||||
),
|
||||
fetchQueryFeaturesStep(
|
||||
config.fetchQueryFeaturesPhase2,
|
||||
MixerPipelineConfig.fetchQueryFeaturesPhase2Step,
|
||||
(previousPipelineResult, executorResult) =>
|
||||
previousPipelineResult.copy(
|
||||
queryFeaturesPhase2 = Some(executorResult),
|
||||
mergedAsyncQueryFeatures = Some(
|
||||
previousPipelineResult.queryFeatures
|
||||
.getOrElse(throw InvalidStepStateException(
|
||||
MixerPipelineConfig.fetchQueryFeaturesPhase2Step,
|
||||
"QueryFeatures"))
|
||||
.asyncFeatureMap ++ executorResult.asyncFeatureMap)
|
||||
),
|
||||
context
|
||||
),
|
||||
asyncFeaturesStep(MixerPipelineConfig.candidatePipelinesStep, context),
|
||||
candidatePipelinesStep(
|
||||
candidatePipelines,
|
||||
config.defaultFailOpenPolicy,
|
||||
config.failOpenPolicies,
|
||||
qualityFactorObserverByPipeline,
|
||||
context),
|
||||
asyncFeaturesStep(MixerPipelineConfig.dependentCandidatePipelinesStep, context),
|
||||
dependentCandidatePipelinesStep(
|
||||
dependentCandidatePipelines,
|
||||
config.defaultFailOpenPolicy,
|
||||
config.failOpenPolicies,
|
||||
qualityFactorObserverByPipeline,
|
||||
context),
|
||||
asyncFeaturesStep(MixerPipelineConfig.resultSelectorsStep, context),
|
||||
resultSelectorsStep(config.resultSelectors, context),
|
||||
domainMarshallingStep(config.domainMarshaller, context),
|
||||
asyncFeaturesStep(MixerPipelineConfig.resultSideEffectsStep, context),
|
||||
resultSideEffectsStep(config.resultSideEffects, context),
|
||||
transportMarshallingStep(config.transportMarshaller, context)
|
||||
)
|
||||
|
||||
val finalArrow = buildCombinedArrowFromSteps(
|
||||
steps = builtSteps,
|
||||
context = context,
|
||||
initialEmptyResult = MixerPipelineResult.empty,
|
||||
stepsInOrderFromConfig = MixerPipelineConfig.stepsInOrder
|
||||
)
|
||||
|
||||
val configFromBuilder = config
|
||||
new MixerPipeline[Query, Result] {
|
||||
override private[core] val config: MixerPipelineConfig[Query, _, Result] = configFromBuilder
|
||||
override val arrow: Arrow[Query, MixerPipelineResult[Result]] = finalArrow
|
||||
override val identifier: MixerPipelineIdentifier = pipelineIdentifier
|
||||
override val alerts: Seq[Alert] = config.alerts
|
||||
override val children: Seq[Component] =
|
||||
config.gates ++
|
||||
config.fetchQueryFeatures ++
|
||||
candidatePipelines ++
|
||||
dependentCandidatePipelines ++
|
||||
config.resultSideEffects ++
|
||||
Seq(config.domainMarshaller, config.transportMarshaller)
|
||||
}
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,49 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.mixer
|
||||
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.product_mixer.core.model.marshalling.HasMarshalling
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineBuilderFactory
|
||||
import com.twitter.product_mixer.core.service.candidate_pipeline_executor.CandidatePipelineExecutor
|
||||
import com.twitter.product_mixer.core.service.domain_marshaller_executor.DomainMarshallerExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.pipeline_result_side_effect_executor.PipelineResultSideEffectExecutor
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutor
|
||||
import com.twitter.product_mixer.core.service.query_feature_hydrator_executor.QueryFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutor
|
||||
import com.twitter.product_mixer.core.service.transport_marshaller_executor.TransportMarshallerExecutor
|
||||
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
|
||||
@Singleton
|
||||
class MixerPipelineBuilderFactory @Inject() (
|
||||
candidatePipelineExecutor: CandidatePipelineExecutor,
|
||||
gateExecutor: GateExecutor,
|
||||
selectorExecutor: SelectorExecutor,
|
||||
queryFeatureHydratorExecutor: QueryFeatureHydratorExecutor,
|
||||
asyncFeatureMapExecutor: AsyncFeatureMapExecutor,
|
||||
domainMarshallerExecutor: DomainMarshallerExecutor,
|
||||
transportMarshallerExecutor: TransportMarshallerExecutor,
|
||||
pipelineResultSideEffectExecutor: PipelineResultSideEffectExecutor,
|
||||
candidatePipelineBuilderFactory: CandidatePipelineBuilderFactory,
|
||||
statsReceiver: StatsReceiver) {
|
||||
def get[
|
||||
Query <: PipelineQuery,
|
||||
DomainResultType <: HasMarshalling,
|
||||
Result
|
||||
]: MixerPipelineBuilder[Query, DomainResultType, Result] = {
|
||||
new MixerPipelineBuilder[Query, DomainResultType, Result](
|
||||
candidatePipelineExecutor,
|
||||
gateExecutor,
|
||||
selectorExecutor,
|
||||
queryFeatureHydratorExecutor,
|
||||
asyncFeatureMapExecutor,
|
||||
domainMarshallerExecutor,
|
||||
transportMarshallerExecutor,
|
||||
pipelineResultSideEffectExecutor,
|
||||
candidatePipelineBuilderFactory,
|
||||
statsReceiver
|
||||
)
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,175 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.mixer
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator
|
||||
import com.twitter.product_mixer.core.functional_component.gate.Gate
|
||||
import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller
|
||||
import com.twitter.product_mixer.core.functional_component.selector.Selector
|
||||
import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect
|
||||
import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.MixerPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.model.marshalling.HasMarshalling
|
||||
import com.twitter.product_mixer.core.pipeline.FailOpenPolicy
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfigCompanion
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.ClosedGate
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.product_mixer.core.quality_factor.QualityFactorConfig
|
||||
|
||||
/**
|
||||
* This is the configuration necessary to generate a Mixer Pipeline. Product code should create a
|
||||
* MixerPipelineConfig, and then use a MixerPipelineBuilder to get the final MixerPipeline which can
|
||||
* process requests.
|
||||
*
|
||||
* @tparam Query - The domain model for the query or request
|
||||
* @tparam UnmarshalledResultType - The result type of the pipeline, but before marshalling to a wire protocol like URT
|
||||
* @tparam Result - The final result that will be served to users
|
||||
*/
|
||||
trait MixerPipelineConfig[Query <: PipelineQuery, UnmarshalledResultType <: HasMarshalling, Result]
|
||||
extends PipelineConfig {
|
||||
|
||||
override val identifier: MixerPipelineIdentifier
|
||||
|
||||
/**
|
||||
* Mixer Pipeline Gates will be executed before any other step (including retrieval from candidate
|
||||
* pipelines). They're executed sequentially, and any "Stop" result will prevent pipeline execution.
|
||||
*/
|
||||
def gates: Seq[Gate[Query]] = Seq.empty
|
||||
|
||||
/**
|
||||
* A mixer pipeline can fetch query-level features before candidate pipelines are executed.
|
||||
*/
|
||||
def fetchQueryFeatures: Seq[QueryFeatureHydrator[Query]] = Seq.empty
|
||||
|
||||
/**
|
||||
* For query-level features that are dependent on query-level features from [[fetchQueryFeatures]]
|
||||
*/
|
||||
def fetchQueryFeaturesPhase2: Seq[QueryFeatureHydrator[Query]] = Seq.empty
|
||||
|
||||
/**
|
||||
* Candidate pipelines retrieve candidates for possible inclusion in the result
|
||||
*/
|
||||
def candidatePipelines: Seq[CandidatePipelineConfig[Query, _, _, _]]
|
||||
|
||||
/**
|
||||
* Dependent candidate pipelines to retrieve candidates that depend on the result of [[candidatePipelines]]
|
||||
* [[DependentCandidatePipelineConfig]] have access to the list of previously retrieved & decorated
|
||||
* candidates for use in constructing the query object.
|
||||
*/
|
||||
def dependentCandidatePipelines: Seq[DependentCandidatePipelineConfig[Query, _, _, _]] = Seq.empty
|
||||
|
||||
/**
|
||||
* [[defaultFailOpenPolicy]] is the [[FailOpenPolicy]] that will be applied to any candidate
|
||||
* pipeline that isn't in the [[failOpenPolicies]] map. By default Candidate Pipelines will fail
|
||||
* open for Closed Gates only.
|
||||
*/
|
||||
def defaultFailOpenPolicy: FailOpenPolicy = FailOpenPolicy(Set(ClosedGate))
|
||||
|
||||
/**
|
||||
* [[failOpenPolicies]] associates [[FailOpenPolicy]]s to specific candidate pipelines using
|
||||
* [[CandidatePipelineIdentifier]].
|
||||
*
|
||||
* @note these [[FailOpenPolicy]]s override the [[defaultFailOpenPolicy]] for a mapped
|
||||
* Candidate Pipeline.
|
||||
*/
|
||||
def failOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy] = Map.empty
|
||||
|
||||
/**
|
||||
** [[qualityFactorConfigs]] associates [[QualityFactorConfig]]s to specific candidate pipelines
|
||||
* using [[CandidatePipelineIdentifier]].
|
||||
*/
|
||||
def qualityFactorConfigs: Map[CandidatePipelineIdentifier, QualityFactorConfig] =
|
||||
Map.empty
|
||||
|
||||
/**
|
||||
* Selectors are executed in sequential order to combine the candidates into a result
|
||||
*/
|
||||
def resultSelectors: Seq[Selector[Query]]
|
||||
|
||||
/**
|
||||
* Mixer result side effects that are executed after selection and domain marshalling
|
||||
*/
|
||||
def resultSideEffects: Seq[PipelineResultSideEffect[Query, UnmarshalledResultType]] = Seq()
|
||||
|
||||
/**
|
||||
* Domain marshaller transforms the selections into the model expected by the marshaller
|
||||
*/
|
||||
def domainMarshaller: DomainMarshaller[Query, UnmarshalledResultType]
|
||||
|
||||
/**
|
||||
* Transport marshaller transforms the model into our line-level API like URT or JSON
|
||||
*/
|
||||
def transportMarshaller: TransportMarshaller[UnmarshalledResultType, Result]
|
||||
|
||||
/**
|
||||
* A pipeline can define a partial function to rescue failures here. They will be treated as failures
|
||||
* from a monitoring standpoint, and cancellation exceptions will always be propagated (they cannot be caught here).
|
||||
*/
|
||||
def failureClassifier: PartialFunction[Throwable, PipelineFailure] = PartialFunction.empty
|
||||
|
||||
/**
|
||||
* Alert can be used to indicate the pipeline's service level objectives. Alerts and
|
||||
* dashboards will be automatically created based on this information.
|
||||
*/
|
||||
val alerts: Seq[Alert] = Seq.empty
|
||||
|
||||
/**
|
||||
* This method is used by the product mixer framework to build the pipeline.
|
||||
*/
|
||||
private[core] final def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
builder: MixerPipelineBuilderFactory
|
||||
): MixerPipeline[Query, Result] =
|
||||
builder.get.build(parentComponentIdentifierStack, this)
|
||||
}
|
||||
|
||||
object MixerPipelineConfig extends PipelineConfigCompanion {
|
||||
val qualityFactorStep: PipelineStepIdentifier = PipelineStepIdentifier("QualityFactor")
|
||||
val gatesStep: PipelineStepIdentifier = PipelineStepIdentifier("Gates")
|
||||
val fetchQueryFeaturesStep: PipelineStepIdentifier = PipelineStepIdentifier("FetchQueryFeatures")
|
||||
val fetchQueryFeaturesPhase2Step: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("FetchQueryFeaturesPhase2")
|
||||
val candidatePipelinesStep: PipelineStepIdentifier = PipelineStepIdentifier("CandidatePipelines")
|
||||
val dependentCandidatePipelinesStep: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("DependentCandidatePipelines")
|
||||
val resultSelectorsStep: PipelineStepIdentifier = PipelineStepIdentifier("ResultSelectors")
|
||||
val domainMarshallerStep: PipelineStepIdentifier = PipelineStepIdentifier("DomainMarshaller")
|
||||
val resultSideEffectsStep: PipelineStepIdentifier = PipelineStepIdentifier("ResultSideEffects")
|
||||
val transportMarshallerStep: PipelineStepIdentifier = PipelineStepIdentifier(
|
||||
"TransportMarshaller")
|
||||
|
||||
/** All the Steps which are executed by a [[MixerPipeline]] in the order in which they are run */
|
||||
override val stepsInOrder: Seq[PipelineStepIdentifier] = Seq(
|
||||
qualityFactorStep,
|
||||
gatesStep,
|
||||
fetchQueryFeaturesStep,
|
||||
fetchQueryFeaturesPhase2Step,
|
||||
asyncFeaturesStep(candidatePipelinesStep),
|
||||
candidatePipelinesStep,
|
||||
asyncFeaturesStep(dependentCandidatePipelinesStep),
|
||||
dependentCandidatePipelinesStep,
|
||||
asyncFeaturesStep(resultSelectorsStep),
|
||||
resultSelectorsStep,
|
||||
domainMarshallerStep,
|
||||
asyncFeaturesStep(resultSideEffectsStep),
|
||||
resultSideEffectsStep,
|
||||
transportMarshallerStep
|
||||
)
|
||||
|
||||
/**
|
||||
* All the Steps which an [[com.twitter.product_mixer.core.functional_component.feature_hydrator.AsyncHydrator AsyncHydrator]]
|
||||
* can be configured to [[com.twitter.product_mixer.core.functional_component.feature_hydrator.AsyncHydrator.hydrateBefore hydrateBefore]]
|
||||
*/
|
||||
override val stepsAsyncFeatureHydrationCanBeCompletedBy: Set[PipelineStepIdentifier] = Set(
|
||||
candidatePipelinesStep,
|
||||
dependentCandidatePipelinesStep,
|
||||
resultSelectorsStep,
|
||||
resultSideEffectsStep
|
||||
)
|
||||
}
|
Binary file not shown.
@ -1,70 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.mixer
|
||||
|
||||
import com.twitter.product_mixer.core.feature.featuremap.asyncfeaturemap.AsyncFeatureMap
|
||||
import com.twitter.product_mixer.core.model.marshalling.HasMarshalling
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineResult
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutorResults
|
||||
import com.twitter.product_mixer.core.service.candidate_pipeline_executor.CandidatePipelineExecutorResult
|
||||
import com.twitter.product_mixer.core.service.domain_marshaller_executor.DomainMarshallerExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.pipeline_result_side_effect_executor.PipelineResultSideEffectExecutor
|
||||
import com.twitter.product_mixer.core.service.quality_factor_executor.QualityFactorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.query_feature_hydrator_executor.QueryFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.transport_marshaller_executor.TransportMarshallerExecutor
|
||||
|
||||
/**
|
||||
* A [[MixerPipelineResult]] includes both the user-visible [[PipelineResult]] and all the
|
||||
* Execution details possible - intermediate results, what components did, etc.
|
||||
*/
|
||||
case class MixerPipelineResult[Result](
|
||||
qualityFactorResult: Option[QualityFactorExecutorResult],
|
||||
gateResult: Option[GateExecutorResult],
|
||||
queryFeatures: Option[QueryFeatureHydratorExecutor.Result],
|
||||
queryFeaturesPhase2: Option[QueryFeatureHydratorExecutor.Result],
|
||||
mergedAsyncQueryFeatures: Option[AsyncFeatureMap],
|
||||
candidatePipelineResults: Option[CandidatePipelineExecutorResult],
|
||||
dependentCandidatePipelineResults: Option[CandidatePipelineExecutorResult],
|
||||
resultSelectorResults: Option[SelectorExecutorResult],
|
||||
domainMarshallerResults: Option[DomainMarshallerExecutor.Result[HasMarshalling]],
|
||||
resultSideEffectResults: Option[PipelineResultSideEffectExecutor.Result],
|
||||
asyncFeatureHydrationResults: Option[AsyncFeatureMapExecutorResults],
|
||||
transportMarshallerResults: Option[TransportMarshallerExecutor.Result[Result]],
|
||||
failure: Option[PipelineFailure],
|
||||
result: Option[Result])
|
||||
extends PipelineResult[Result] {
|
||||
|
||||
override def withFailure(failure: PipelineFailure): PipelineResult[Result] =
|
||||
copy(failure = Some(failure))
|
||||
|
||||
override def withResult(result: Result): PipelineResult[Result] = copy(result = Some(result))
|
||||
|
||||
/**
|
||||
* resultSize is calculated based on the selector results rather than the marshalled results. The
|
||||
* structure of the marshalled format is unknown, making operating on selector results more
|
||||
* convenient. This will implicitly excluded cursors built during marshalling but cursors don't
|
||||
* contribute to the result size anyway.
|
||||
*/
|
||||
override val resultSize: Int =
|
||||
resultSelectorResults.map(_.selectedCandidates).map(PipelineResult.resultSize).getOrElse(0)
|
||||
}
|
||||
|
||||
object MixerPipelineResult {
|
||||
def empty[A]: MixerPipelineResult[A] = MixerPipelineResult(
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None
|
||||
)
|
||||
}
|
@ -1,14 +0,0 @@
|
||||
scala_library(
|
||||
sources = ["*.scala"],
|
||||
compiler_option_sets = ["fatal_warnings"],
|
||||
platform = "java8",
|
||||
strict_deps = True,
|
||||
tags = ["bazel-compatible"],
|
||||
dependencies = [
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier",
|
||||
"util/util-jackson/src/main/scala/com/twitter/util/jackson",
|
||||
],
|
||||
exports = [
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier",
|
||||
],
|
||||
)
|
Binary file not shown.
Binary file not shown.
@ -1,49 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.pipeline_failure
|
||||
|
||||
import com.fasterxml.jackson.databind.annotation.JsonSerialize
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import scala.util.control.NoStackTrace
|
||||
|
||||
/**
|
||||
* Pipeline Failures represent pipeline requests that were not able to complete.
|
||||
*
|
||||
* A pipeline result will always define either a result or a failure.
|
||||
*
|
||||
* The reason field should not be displayed to end-users, and is free to change over time.
|
||||
* It should always be free of private user data such that we can log it.
|
||||
*
|
||||
* The pipeline can classify it's own failures into categories (timeouts, invalid arguments,
|
||||
* rate limited, etc) such that the caller can choose how to handle it.
|
||||
*
|
||||
* @note [[componentStack]] should only be set by the product mixer framework,
|
||||
* it should **NOT** be set when making a [[PipelineFailure]]
|
||||
*/
|
||||
@JsonSerialize(using = classOf[PipelineFailureSerializer])
|
||||
case class PipelineFailure(
|
||||
category: PipelineFailureCategory,
|
||||
reason: String,
|
||||
underlying: Option[Throwable] = None,
|
||||
componentStack: Option[ComponentIdentifierStack] = None)
|
||||
extends Exception(
|
||||
"PipelineFailure(" +
|
||||
s"category = $category, " +
|
||||
s"reason = $reason, " +
|
||||
s"underlying = $underlying, " +
|
||||
s"componentStack = $componentStack)",
|
||||
underlying.orNull
|
||||
) {
|
||||
override def toString: String = getMessage
|
||||
|
||||
/** Returns an updated copy of this [[PipelineFailure]] with the same exception stacktrace */
|
||||
def copy(
|
||||
category: PipelineFailureCategory = this.category,
|
||||
reason: String = this.reason,
|
||||
underlying: Option[Throwable] = this.underlying,
|
||||
componentStack: Option[ComponentIdentifierStack] = this.componentStack
|
||||
): PipelineFailure = {
|
||||
val newPipelineFailure =
|
||||
new PipelineFailure(category, reason, underlying, componentStack) with NoStackTrace
|
||||
newPipelineFailure.setStackTrace(this.getStackTrace)
|
||||
newPipelineFailure
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,190 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.pipeline_failure
|
||||
|
||||
/**
|
||||
* Failures are grouped into categories based on which party is 'responsible' for the issue. This
|
||||
* is important for generating accurate SLOs and ensuring that the correct team is alerted.
|
||||
*/
|
||||
sealed trait PipelineFailureCategory {
|
||||
val categoryName: String
|
||||
val failureName: String
|
||||
}
|
||||
|
||||
/**
|
||||
* Client Failures are failures where the client is deemed responsible for the issue. Such as by
|
||||
* issuing an invalid request or not having the right permissions.
|
||||
*
|
||||
* A failure might belong in this category if it relates to behaviour on the client and is not
|
||||
* actionable by the team which owns the product.
|
||||
*/
|
||||
trait ClientFailure extends PipelineFailureCategory {
|
||||
override val categoryName: String = "ClientFailure"
|
||||
}
|
||||
|
||||
/**
|
||||
* The requested product is disabled so the request cannot be served.
|
||||
*/
|
||||
case object ProductDisabled extends ClientFailure {
|
||||
override val failureName: String = "ProductDisabled"
|
||||
}
|
||||
|
||||
/**
|
||||
* The request was deemed invalid by this or a backing service.
|
||||
*/
|
||||
case object BadRequest extends ClientFailure {
|
||||
override val failureName: String = "BadRequest"
|
||||
}
|
||||
|
||||
/**
|
||||
* Credentials proving the identity of the caller were missing, not trusted, or expired.
|
||||
* For example, an auth cookie might be expired and in need of refreshing.
|
||||
*
|
||||
* Do not confuse this with Authorization, where the credentials are believed but not allowed to perform the operation.
|
||||
*/
|
||||
case object Authentication extends ClientFailure {
|
||||
override val failureName: String = "Authentication"
|
||||
}
|
||||
|
||||
/**
|
||||
* The operation was forbidden (often, but not always, by a Strato access control policy).
|
||||
*
|
||||
* Do not confuse this with Authentication, where the given credentials were missing, not trusted, or expired.
|
||||
*/
|
||||
case object Unauthorized extends ClientFailure {
|
||||
override val failureName: String = "Unauthorized"
|
||||
}
|
||||
|
||||
/**
|
||||
* The operation returned a Not Found response.
|
||||
*/
|
||||
case object NotFound extends ClientFailure {
|
||||
override val failureName: String = "NotFound"
|
||||
}
|
||||
|
||||
/**
|
||||
* An invalid input is included in a cursor field.
|
||||
*/
|
||||
case object MalformedCursor extends ClientFailure {
|
||||
override val failureName: String = "MalformedCursor"
|
||||
}
|
||||
|
||||
/**
|
||||
* The operation did not succeed due to a closed gate
|
||||
*/
|
||||
case object ClosedGate extends ClientFailure {
|
||||
override val failureName: String = "ClosedGate"
|
||||
}
|
||||
|
||||
/**
|
||||
* Server Failures are failures for which the owner of the product is responsible. Typically this
|
||||
* means the request was valid but an issue within Product Mixer or a dependent service prevented
|
||||
* it from succeeding.
|
||||
*
|
||||
* Server Failures contribute to the success rate SLO for the product.
|
||||
*/
|
||||
trait ServerFailure extends PipelineFailureCategory {
|
||||
override val categoryName: String = "ServerFailure"
|
||||
}
|
||||
|
||||
/**
|
||||
* Unclassified failures occur when product code throws an exception that Product Mixer does not
|
||||
* know how to classify.
|
||||
*
|
||||
* They can be used in failOpen policies, etc - but it's always preferred to instead add additional
|
||||
* classification logic and to keep Unclassified failures at 0.
|
||||
*/
|
||||
case object UncategorizedServerFailure extends ServerFailure {
|
||||
override val failureName: String = "UncategorizedServerFailure"
|
||||
}
|
||||
|
||||
/**
|
||||
* A hydrator or transformer returned a misconfigured feature map, this indicates a customer
|
||||
* configuration error. The owner of the component should make sure the hydrator always returns a
|
||||
* [[FeatureMap]] with the all features defined in the hydrator also set in the map, it should not have
|
||||
* any unregistered features nor should registered features be absent.
|
||||
*/
|
||||
case object MisconfiguredFeatureMapFailure extends ServerFailure {
|
||||
override val failureName: String = "MisconfiguredFeatureMapFailure"
|
||||
}
|
||||
|
||||
/**
|
||||
* A PipelineSelector returned an invalid ComponentIdentifier.
|
||||
*
|
||||
* A pipeline selector should choose the identifier of a pipeline that is contained by the 'pipelines'
|
||||
* sequence of the ProductPipelineConfig.
|
||||
*/
|
||||
case object InvalidPipelineSelected extends ServerFailure {
|
||||
override val failureName: String = "InvalidPipelineSelected"
|
||||
}
|
||||
|
||||
/**
|
||||
* Failures that occur when product code reaches an unexpected or otherwise illegal state.
|
||||
*/
|
||||
case object IllegalStateFailure extends ServerFailure {
|
||||
override val failureName: String = "IllegalStateFailure"
|
||||
}
|
||||
|
||||
/**
|
||||
* An unexpected candidate was returned in a candidate source that was unable to be transformed.
|
||||
*/
|
||||
case object UnexpectedCandidateResult extends ServerFailure {
|
||||
override val failureName: String = "UnexpectedCandidateResult"
|
||||
}
|
||||
|
||||
/**
|
||||
* An unexpected Candidate was returned in a marshaller
|
||||
*/
|
||||
case object UnexpectedCandidateInMarshaller extends ServerFailure {
|
||||
override val failureName: String = "UnexpectedCandidateInMarshaller"
|
||||
}
|
||||
|
||||
/**
|
||||
* Pipeline execution failed due to an incorrectly configured quality factor (e.g, accessing
|
||||
* quality factor state for a pipeline that does not have quality factor configured)
|
||||
*/
|
||||
case object MisconfiguredQualityFactor extends ServerFailure {
|
||||
override val failureName: String = "MisconfiguredQualityFactor"
|
||||
}
|
||||
|
||||
/**
|
||||
* Pipeline execution failed due to an incorrectly configured decorator (e.g, decorator
|
||||
* returned the wrong type or tried to decorate an already decorated candidate)
|
||||
*/
|
||||
case object MisconfiguredDecorator extends ServerFailure {
|
||||
override val failureName: String = "MisconfiguredDecorator"
|
||||
}
|
||||
|
||||
/**
|
||||
* Candidate Source Pipeline execution failed due to a timeout.
|
||||
*/
|
||||
case object CandidateSourceTimeout extends ServerFailure {
|
||||
override val failureName: String = "CandidateSourceTimeout"
|
||||
}
|
||||
|
||||
/**
|
||||
* Platform Failures are issues in the core Product Mixer logic itself which prevent a pipeline from
|
||||
* properly executing. These failures are the responsibility of the Product Mixer team.
|
||||
*/
|
||||
trait PlatformFailure extends PipelineFailureCategory {
|
||||
override val categoryName: String = "PlatformFailure"
|
||||
}
|
||||
|
||||
/**
|
||||
* Pipeline execution failed due to an unexpected error in Product Mixer.
|
||||
*
|
||||
* ExecutionFailed indicates a bug with the core Product Mixer execution logic rather than with a
|
||||
* specific product. For example, a bug in PipelineBuilder leading to us returning a
|
||||
* ProductPipelineResult that neither succeeded nor failed.
|
||||
*/
|
||||
case object ExecutionFailed extends PlatformFailure {
|
||||
override val failureName: String = "ExecutionFailed"
|
||||
}
|
||||
|
||||
/**
|
||||
* Pipeline execution failed due to a feature hydration failure.
|
||||
*
|
||||
* FeatureHydrationFailed indicates that the underlying hydration for a feature defined in a hydrator
|
||||
* failed (e.g, typically from a RPC call failing).
|
||||
*/
|
||||
case object FeatureHydrationFailed extends PlatformFailure {
|
||||
override val failureName: String = "FeatureHydrationFailed"
|
||||
}
|
Binary file not shown.
@ -1,13 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.pipeline_failure
|
||||
|
||||
/** Represents a way to classify a given [[Throwable]] to a [[PipelineFailure]] */
|
||||
case class PipelineFailureClassifier(
|
||||
classifier: PartialFunction[Throwable, PipelineFailure])
|
||||
extends PartialFunction[Throwable, PipelineFailure] {
|
||||
override def isDefinedAt(throwable: Throwable): Boolean = classifier.isDefinedAt(throwable)
|
||||
override def apply(throwable: Throwable): PipelineFailure = classifier.apply(throwable)
|
||||
}
|
||||
|
||||
private[core] object PipelineFailureClassifier {
|
||||
val Empty: PipelineFailureClassifier = PipelineFailureClassifier(PartialFunction.empty)
|
||||
}
|
Binary file not shown.
@ -1,67 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.pipeline_failure
|
||||
|
||||
import com.fasterxml.jackson.core.JsonGenerator
|
||||
import com.fasterxml.jackson.databind.JsonSerializer
|
||||
import com.fasterxml.jackson.databind.SerializerProvider
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
|
||||
private[pipeline_failure] class PipelineFailureSerializer()
|
||||
extends JsonSerializer[PipelineFailure] {
|
||||
|
||||
private sealed trait BaseSerializableException
|
||||
|
||||
private case class SerializableException(
|
||||
`class`: String,
|
||||
message: String,
|
||||
stackTrace: Seq[String],
|
||||
cause: Option[BaseSerializableException])
|
||||
extends BaseSerializableException
|
||||
|
||||
private case class SerializablePipelineFailure(
|
||||
category: String,
|
||||
reason: String,
|
||||
underlying: Option[BaseSerializableException],
|
||||
componentStack: Option[ComponentIdentifierStack],
|
||||
stackTrace: Seq[String])
|
||||
extends BaseSerializableException
|
||||
|
||||
private def serializeStackTrace(stackTrace: Array[StackTraceElement]): Seq[String] =
|
||||
stackTrace.map(stackTraceElement => "at " + stackTraceElement.toString)
|
||||
|
||||
private def mkSerializableException(
|
||||
t: Throwable,
|
||||
recursionDepth: Int = 0
|
||||
): Option[BaseSerializableException] = {
|
||||
t match {
|
||||
case _ if recursionDepth > 4 =>
|
||||
// in the unfortunate case of a super deep chain of exceptions, stop if we get too deep
|
||||
None
|
||||
case pipelineFailure: PipelineFailure =>
|
||||
Some(
|
||||
SerializablePipelineFailure(
|
||||
category =
|
||||
pipelineFailure.category.categoryName + "/" + pipelineFailure.category.failureName,
|
||||
reason = pipelineFailure.reason,
|
||||
underlying =
|
||||
pipelineFailure.underlying.flatMap(mkSerializableException(_, recursionDepth + 1)),
|
||||
componentStack = pipelineFailure.componentStack,
|
||||
stackTrace = serializeStackTrace(pipelineFailure.getStackTrace)
|
||||
))
|
||||
case t =>
|
||||
Some(
|
||||
SerializableException(
|
||||
`class` = t.getClass.getName,
|
||||
message = t.getMessage,
|
||||
stackTrace = serializeStackTrace(t.getStackTrace),
|
||||
cause = Option(t.getCause).flatMap(mkSerializableException(_, recursionDepth + 1))
|
||||
)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
override def serialize(
|
||||
pipelineFailure: PipelineFailure,
|
||||
gen: JsonGenerator,
|
||||
serializers: SerializerProvider
|
||||
): Unit = serializers.defaultSerializeValue(mkSerializableException(pipelineFailure), gen)
|
||||
}
|
@ -1,51 +0,0 @@
|
||||
scala_library(
|
||||
sources = ["*.scala"],
|
||||
compiler_option_sets = ["fatal_warnings"],
|
||||
platform = "java8",
|
||||
strict_deps = True,
|
||||
tags = ["bazel-compatible"],
|
||||
dependencies = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"configapi/configapi-core",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/access_policy",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_execution_logger",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_selector_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/quality_factor_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util",
|
||||
"stitch/stitch-core",
|
||||
"stringcenter/client/src/main/scala/com/twitter/stringcenter/client",
|
||||
"stringcenter/client/src/main/scala/com/twitter/stringcenter/client/stitch",
|
||||
],
|
||||
exports = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"configapi/configapi-core",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/access_policy",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_execution_logger",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_selector_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/quality_factor_executor",
|
||||
"stitch/stitch-core",
|
||||
],
|
||||
)
|
Binary file not shown.
Binary file not shown.
@ -1,28 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.product
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.common.access_policy.WithDebugAccessPolicies
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ProductPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.marshalling.request.Request
|
||||
import com.twitter.product_mixer.core.pipeline.Pipeline
|
||||
import com.twitter.stitch.Arrow
|
||||
|
||||
/**
|
||||
* A Product Pipeline
|
||||
*
|
||||
* This is an abstract class, as we only construct these via the [[ProductPipelineBuilder]].
|
||||
*
|
||||
* A [[ProductPipeline]] is capable of processing a [[Request]] and returning a response.
|
||||
*
|
||||
* @tparam RequestType the domain model for the query or request
|
||||
* @tparam ResponseType the final marshalled result type
|
||||
*/
|
||||
abstract class ProductPipeline[RequestType <: Request, ResponseType] private[product]
|
||||
extends Pipeline[ProductPipelineRequest[RequestType], ResponseType]
|
||||
with WithDebugAccessPolicies {
|
||||
override private[core] val config: ProductPipelineConfig[RequestType, _, ResponseType]
|
||||
override val arrow: Arrow[
|
||||
ProductPipelineRequest[RequestType],
|
||||
ProductPipelineResult[ResponseType]
|
||||
]
|
||||
override val identifier: ProductPipelineIdentifier
|
||||
}
|
Binary file not shown.
@ -1,385 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.product
|
||||
|
||||
import com.twitter.finagle.mtls.authentication.ServiceIdentifier
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.finagle.tracing.Trace
|
||||
import com.twitter.finagle.transport.Transport
|
||||
import com.twitter.product_mixer.core.functional_component.common.access_policy.AccessPolicy
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.gate.Gate
|
||||
import com.twitter.product_mixer.core.gate.DenyLoggedOutUsersGate
|
||||
import com.twitter.product_mixer.core.gate.ParamGate
|
||||
import com.twitter.product_mixer.core.gate.ParamGate.EnabledGateSuffix
|
||||
import com.twitter.product_mixer.core.gate.ParamGate.SupportedClientGateSuffix
|
||||
import com.twitter.product_mixer.core.model.common.Component
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ProductPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.model.marshalling.request.Request
|
||||
import com.twitter.product_mixer.core.pipeline.InvalidStepStateException
|
||||
import com.twitter.product_mixer.core.pipeline.Pipeline
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineBuilder
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineBuilderFactory
|
||||
import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineResult
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailureClassifier
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.ProductDisabled
|
||||
import com.twitter.product_mixer.core.pipeline.recommendation.RecommendationPipelineBuilderFactory
|
||||
import com.twitter.product_mixer.core.pipeline.recommendation.RecommendationPipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.recommendation.RecommendationPipelineResult
|
||||
import com.twitter.product_mixer.core.quality_factor.HasQualityFactorStatus
|
||||
import com.twitter.product_mixer.core.quality_factor.QualityFactorObserver
|
||||
import com.twitter.product_mixer.core.quality_factor.QualityFactorStatus
|
||||
import com.twitter.product_mixer.core.service.Executor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.StoppedGateException
|
||||
import com.twitter.product_mixer.core.service.pipeline_execution_logger.PipelineExecutionLogger
|
||||
import com.twitter.product_mixer.core.service.pipeline_executor.PipelineExecutor
|
||||
import com.twitter.product_mixer.core.service.pipeline_executor.PipelineExecutorRequest
|
||||
import com.twitter.product_mixer.core.service.pipeline_executor.PipelineExecutorResult
|
||||
import com.twitter.product_mixer.core.service.pipeline_selector_executor.PipelineSelectorExecutor
|
||||
import com.twitter.product_mixer.core.service.pipeline_selector_executor.PipelineSelectorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.quality_factor_executor.QualityFactorExecutorResult
|
||||
import com.twitter.stitch.Arrow
|
||||
import com.twitter.stringcenter.client.StringCenterRequestContext
|
||||
import com.twitter.stringcenter.client.stitch.StringCenterRequestContextLetter
|
||||
import com.twitter.timelines.configapi.Params
|
||||
import com.twitter.util.logging.Logging
|
||||
import org.slf4j.MDC
|
||||
|
||||
class ProductPipelineBuilder[TRequest <: Request, Query <: PipelineQuery, Response](
|
||||
gateExecutor: GateExecutor,
|
||||
pipelineSelectorExecutor: PipelineSelectorExecutor,
|
||||
pipelineExecutor: PipelineExecutor,
|
||||
mixerPipelineBuilderFactory: MixerPipelineBuilderFactory,
|
||||
recommendationPipelineBuilderFactory: RecommendationPipelineBuilderFactory,
|
||||
override val statsReceiver: StatsReceiver,
|
||||
pipelineExecutionLogger: PipelineExecutionLogger)
|
||||
extends PipelineBuilder[ProductPipelineRequest[TRequest]]
|
||||
with Logging { builder =>
|
||||
|
||||
override type UnderlyingResultType = Response
|
||||
override type PipelineResultType = ProductPipelineResult[Response]
|
||||
|
||||
/**
|
||||
* Query Transformer Step is implemented inline instead of using an executor.
|
||||
*
|
||||
* It's a simple, synchronous step that executes the query transformer.
|
||||
*
|
||||
* Since the output of the transformer is used in multiple other steps (Gate, Pipeline Execution),
|
||||
* we've promoted the transformer to a step so that it's outputs can be reused easily.
|
||||
*/
|
||||
def pipelineQueryTransformerStep(
|
||||
queryTransformer: (TRequest, Params) => Query,
|
||||
context: Executor.Context
|
||||
): Step[ProductPipelineRequest[TRequest], Query] =
|
||||
new Step[ProductPipelineRequest[TRequest], Query] {
|
||||
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
ProductPipelineConfig.pipelineQueryTransformerStep
|
||||
|
||||
override def executorArrow: Arrow[ProductPipelineRequest[TRequest], Query] = {
|
||||
wrapWithErrorHandling(context, identifier)(
|
||||
Arrow.map[ProductPipelineRequest[TRequest], Query] {
|
||||
case ProductPipelineRequest(request, params) => queryTransformer(request, params)
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ProductPipelineRequest[TRequest],
|
||||
previousResult: ProductPipelineResult[Response]
|
||||
): ProductPipelineRequest[TRequest] = query
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ProductPipelineResult[Response],
|
||||
executorResult: Query
|
||||
): ProductPipelineResult[Response] =
|
||||
previousPipelineResult.copy(transformedQuery = Some(executorResult))
|
||||
}
|
||||
|
||||
def qualityFactorStep(
|
||||
qualityFactorStatus: QualityFactorStatus
|
||||
): Step[Query, QualityFactorExecutorResult] = {
|
||||
new Step[Query, QualityFactorExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = ProductPipelineConfig.qualityFactorStep
|
||||
|
||||
override def executorArrow: Arrow[Query, QualityFactorExecutorResult] =
|
||||
Arrow
|
||||
.map[Query, QualityFactorExecutorResult] { _ =>
|
||||
QualityFactorExecutorResult(
|
||||
pipelineQualityFactors =
|
||||
qualityFactorStatus.qualityFactorByPipeline.mapValues(_.currentValue)
|
||||
)
|
||||
}
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ProductPipelineRequest[TRequest],
|
||||
previousResult: ProductPipelineResult[Response]
|
||||
): Query = previousResult.transformedQuery
|
||||
.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "TransformedQuery")
|
||||
}.asInstanceOf[Query]
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ProductPipelineResult[Response],
|
||||
executorResult: QualityFactorExecutorResult
|
||||
): ProductPipelineResult[Response] = {
|
||||
previousPipelineResult.copy(
|
||||
transformedQuery = previousPipelineResult.transformedQuery.map {
|
||||
case queryWithQualityFactor: HasQualityFactorStatus =>
|
||||
queryWithQualityFactor
|
||||
.withQualityFactorStatus(qualityFactorStatus).asInstanceOf[Query]
|
||||
case query =>
|
||||
query
|
||||
},
|
||||
qualityFactorResult = Some(executorResult)
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
def gatesStep(
|
||||
gates: Seq[Gate[Query]],
|
||||
context: Executor.Context
|
||||
): Step[Query, GateExecutorResult] = new Step[Query, GateExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = ProductPipelineConfig.gatesStep
|
||||
|
||||
override def executorArrow: Arrow[Query, GateExecutorResult] = {
|
||||
gateExecutor.arrow(gates, context)
|
||||
}
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ProductPipelineRequest[TRequest],
|
||||
previousResult: ProductPipelineResult[Response]
|
||||
): Query = previousResult.transformedQuery
|
||||
.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "TransformedQuery")
|
||||
}.asInstanceOf[Query]
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ProductPipelineResult[Response],
|
||||
executorResult: GateExecutorResult
|
||||
): ProductPipelineResult[Response] =
|
||||
previousPipelineResult.copy(gateResult = Some(executorResult))
|
||||
}
|
||||
|
||||
def pipelineSelectorStep(
|
||||
pipelineByIdentifer: Map[ComponentIdentifier, Pipeline[Query, Response]],
|
||||
pipelineSelector: Query => ComponentIdentifier,
|
||||
context: Executor.Context
|
||||
): Step[Query, PipelineSelectorExecutorResult] =
|
||||
new Step[Query, PipelineSelectorExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = ProductPipelineConfig.pipelineSelectorStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
Query,
|
||||
PipelineSelectorExecutorResult
|
||||
] = pipelineSelectorExecutor.arrow(pipelineByIdentifer, pipelineSelector, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ProductPipelineRequest[TRequest],
|
||||
previousResult: ProductPipelineResult[Response]
|
||||
): Query =
|
||||
previousResult.transformedQuery
|
||||
.getOrElse(throw InvalidStepStateException(identifier, "TransformedQuery")).asInstanceOf[
|
||||
Query]
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ProductPipelineResult[Response],
|
||||
executorResult: PipelineSelectorExecutorResult
|
||||
): ProductPipelineResult[Response] =
|
||||
previousPipelineResult.copy(pipelineSelectorResult = Some(executorResult))
|
||||
}
|
||||
|
||||
def pipelineExecutionStep(
|
||||
pipelineByIdentifier: Map[ComponentIdentifier, Pipeline[Query, Response]],
|
||||
qualityFactorObserverByPipeline: Map[ComponentIdentifier, QualityFactorObserver],
|
||||
context: Executor.Context
|
||||
): Step[PipelineExecutorRequest[Query], PipelineExecutorResult[Response]] =
|
||||
new Step[PipelineExecutorRequest[Query], PipelineExecutorResult[Response]] {
|
||||
override def identifier: PipelineStepIdentifier = ProductPipelineConfig.pipelineExecutionStep
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
PipelineExecutorRequest[Query],
|
||||
PipelineExecutorResult[Response]
|
||||
] = {
|
||||
pipelineExecutor.arrow(pipelineByIdentifier, qualityFactorObserverByPipeline, context)
|
||||
}
|
||||
|
||||
override def inputAdaptor(
|
||||
request: ProductPipelineRequest[TRequest],
|
||||
previousResult: ProductPipelineResult[Response]
|
||||
): PipelineExecutorRequest[Query] = {
|
||||
val query = previousResult.transformedQuery
|
||||
.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "TransformedQuery")
|
||||
}.asInstanceOf[Query]
|
||||
|
||||
val pipelineIdentifier = previousResult.pipelineSelectorResult
|
||||
.map(_.pipelineIdentifier).getOrElse {
|
||||
throw InvalidStepStateException(identifier, "PipelineSelectorResult")
|
||||
}
|
||||
|
||||
PipelineExecutorRequest(query, pipelineIdentifier)
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ProductPipelineResult[Response],
|
||||
executorResult: PipelineExecutorResult[Response]
|
||||
): ProductPipelineResult[Response] = {
|
||||
|
||||
val mixerPipelineResult = executorResult.pipelineResult match {
|
||||
case mixerPipelineResult: MixerPipelineResult[Response] @unchecked =>
|
||||
Some(mixerPipelineResult)
|
||||
case _ =>
|
||||
None
|
||||
}
|
||||
|
||||
val recommendationPipelineResult = executorResult.pipelineResult match {
|
||||
case recommendationPipelineResult: RecommendationPipelineResult[
|
||||
_,
|
||||
Response
|
||||
] @unchecked =>
|
||||
Some(recommendationPipelineResult)
|
||||
case _ =>
|
||||
None
|
||||
}
|
||||
|
||||
previousPipelineResult.copy(
|
||||
mixerPipelineResult = mixerPipelineResult,
|
||||
recommendationPipelineResult = recommendationPipelineResult,
|
||||
traceId = Trace.idOption.map(_.traceId.toString()),
|
||||
result = executorResult.pipelineResult.result
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
config: ProductPipelineConfig[TRequest, Query, Response]
|
||||
): ProductPipeline[TRequest, Response] = {
|
||||
|
||||
val pipelineIdentifier = config.identifier
|
||||
|
||||
val context = Executor.Context(
|
||||
PipelineFailureClassifier(
|
||||
config.failureClassifier.orElse(StoppedGateException.classifier(ProductDisabled))),
|
||||
parentComponentIdentifierStack.push(pipelineIdentifier)
|
||||
)
|
||||
|
||||
val denyLoggedOutUsersGate = if (config.denyLoggedOutUsers) {
|
||||
Some(DenyLoggedOutUsersGate(pipelineIdentifier))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
val enabledGate: ParamGate =
|
||||
ParamGate(pipelineIdentifier + EnabledGateSuffix, config.paramConfig.EnabledDeciderParam)
|
||||
val supportedClientGate =
|
||||
ParamGate(
|
||||
pipelineIdentifier + SupportedClientGateSuffix,
|
||||
config.paramConfig.SupportedClientParam)
|
||||
|
||||
/**
|
||||
* Evaluate enabled decider gate first since if it's off, there is no reason to proceed
|
||||
* Next evaluate supported client feature switch gate, followed by customer configured gates
|
||||
*/
|
||||
val allGates =
|
||||
denyLoggedOutUsersGate.toSeq ++: enabledGate +: supportedClientGate +: config.gates
|
||||
|
||||
val childPipelines: Seq[Pipeline[Query, Response]] =
|
||||
config.pipelines.map {
|
||||
case mixerConfig: MixerPipelineConfig[Query, _, Response] =>
|
||||
mixerConfig.build(context.componentStack, mixerPipelineBuilderFactory)
|
||||
case recommendationConfig: RecommendationPipelineConfig[Query, _, _, Response] =>
|
||||
recommendationConfig.build(context.componentStack, recommendationPipelineBuilderFactory)
|
||||
case other =>
|
||||
throw new IllegalArgumentException(
|
||||
s"Product Pipelines only support Mixer and Recommendation pipelines, not $other")
|
||||
}
|
||||
|
||||
val pipelineByIdentifier: Map[ComponentIdentifier, Pipeline[Query, Response]] =
|
||||
childPipelines.map { pipeline =>
|
||||
(pipeline.identifier, pipeline)
|
||||
}.toMap
|
||||
|
||||
val qualityFactorStatus: QualityFactorStatus =
|
||||
QualityFactorStatus.build(config.qualityFactorConfigs)
|
||||
|
||||
val qualityFactorObserverByPipeline = qualityFactorStatus.qualityFactorByPipeline.mapValues {
|
||||
qualityFactor =>
|
||||
qualityFactor.buildObserver()
|
||||
}
|
||||
|
||||
buildGaugesForQualityFactor(pipelineIdentifier, qualityFactorStatus, statsReceiver)
|
||||
|
||||
/**
|
||||
* Initialize MDC with access logging with everything we have at request time. We can put
|
||||
* more stuff into MDC later down the pipeline, but at risk of exceptions/errors preventing
|
||||
* them from being added
|
||||
*/
|
||||
val mdcInitArrow =
|
||||
Arrow.map[ProductPipelineRequest[TRequest], ProductPipelineRequest[TRequest]] { request =>
|
||||
val serviceIdentifier = ServiceIdentifier.fromCertificate(Transport.peerCertificate)
|
||||
MDC.put("product", config.product.identifier.name)
|
||||
MDC.put("serviceIdentifier", ServiceIdentifier.asString(serviceIdentifier))
|
||||
request
|
||||
}
|
||||
|
||||
val builtSteps = Seq(
|
||||
pipelineQueryTransformerStep(config.pipelineQueryTransformer, context),
|
||||
qualityFactorStep(qualityFactorStatus),
|
||||
gatesStep(allGates, context),
|
||||
pipelineSelectorStep(pipelineByIdentifier, config.pipelineSelector, context),
|
||||
pipelineExecutionStep(pipelineByIdentifier, qualityFactorObserverByPipeline, context)
|
||||
)
|
||||
|
||||
val underlying: Arrow[ProductPipelineRequest[TRequest], ProductPipelineResult[Response]] =
|
||||
buildCombinedArrowFromSteps(
|
||||
steps = builtSteps,
|
||||
context = context,
|
||||
initialEmptyResult = ProductPipelineResult.empty,
|
||||
stepsInOrderFromConfig = ProductPipelineConfig.stepsInOrder
|
||||
)
|
||||
|
||||
/**
|
||||
* Unlike other components and pipelines, [[ProductPipeline]] must be observed in the
|
||||
* [[ProductPipelineBuilder]] directly because the resulting [[ProductPipeline.arrow]]
|
||||
* is run directly without an executor so must contain all stats.
|
||||
*/
|
||||
val observed =
|
||||
wrapProductPipelineWithExecutorBookkeeping[
|
||||
ProductPipelineRequest[TRequest],
|
||||
ProductPipelineResult[Response]
|
||||
](context, pipelineIdentifier)(underlying)
|
||||
|
||||
val finalArrow: Arrow[ProductPipelineRequest[TRequest], ProductPipelineResult[Response]] =
|
||||
Arrow
|
||||
.letWithArg[
|
||||
ProductPipelineRequest[TRequest],
|
||||
ProductPipelineResult[Response],
|
||||
StringCenterRequestContext](StringCenterRequestContextLetter)(request =>
|
||||
StringCenterRequestContext(
|
||||
request.request.clientContext.languageCode,
|
||||
request.request.clientContext.countryCode
|
||||
))(
|
||||
mdcInitArrow
|
||||
.andThen(observed)
|
||||
.onSuccess(result => result.transformedQuery.map(pipelineExecutionLogger(_, result))))
|
||||
|
||||
val configFromBuilder = config
|
||||
new ProductPipeline[TRequest, Response] {
|
||||
override private[core] val config: ProductPipelineConfig[TRequest, _, Response] =
|
||||
configFromBuilder
|
||||
override val arrow: Arrow[ProductPipelineRequest[TRequest], ProductPipelineResult[Response]] =
|
||||
finalArrow
|
||||
override val identifier: ProductPipelineIdentifier = pipelineIdentifier
|
||||
override val alerts: Seq[Alert] = config.alerts
|
||||
override val debugAccessPolicies: Set[AccessPolicy] = config.debugAccessPolicies
|
||||
override val children: Seq[Component] = allGates ++ childPipelines
|
||||
}
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,39 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.product
|
||||
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.product_mixer.core.model.marshalling.request.Request
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineBuilderFactory
|
||||
import com.twitter.product_mixer.core.pipeline.recommendation.RecommendationPipelineBuilderFactory
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.pipeline_execution_logger.PipelineExecutionLogger
|
||||
import com.twitter.product_mixer.core.service.pipeline_executor.PipelineExecutor
|
||||
import com.twitter.product_mixer.core.service.pipeline_selector_executor.PipelineSelectorExecutor
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
|
||||
@Singleton
|
||||
class ProductPipelineBuilderFactory @Inject() (
|
||||
gateExecutor: GateExecutor,
|
||||
pipelineSelectorExecutor: PipelineSelectorExecutor,
|
||||
pipelineExecutor: PipelineExecutor,
|
||||
mixerPipelineBuilderFactory: MixerPipelineBuilderFactory,
|
||||
recommendationPipelineBuilderFactory: RecommendationPipelineBuilderFactory,
|
||||
statsReceiver: StatsReceiver,
|
||||
pipelineExecutionLogger: PipelineExecutionLogger) {
|
||||
def get[
|
||||
TRequest <: Request,
|
||||
Query <: PipelineQuery,
|
||||
Response
|
||||
]: ProductPipelineBuilder[TRequest, Query, Response] = {
|
||||
new ProductPipelineBuilder[TRequest, Query, Response](
|
||||
gateExecutor,
|
||||
pipelineSelectorExecutor,
|
||||
pipelineExecutor,
|
||||
mixerPipelineBuilderFactory,
|
||||
recommendationPipelineBuilderFactory,
|
||||
statsReceiver,
|
||||
pipelineExecutionLogger
|
||||
)
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,107 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.product
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.common.access_policy.AccessPolicy
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.gate.Gate
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ProductPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.model.marshalling.request.Product
|
||||
import com.twitter.product_mixer.core.model.marshalling.request.Request
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfigCompanion
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.product_mixer.core.product.ProductParamConfig
|
||||
import com.twitter.product_mixer.core.quality_factor.QualityFactorConfig
|
||||
import com.twitter.timelines.configapi.Params
|
||||
|
||||
trait ProductPipelineConfig[TRequest <: Request, Query <: PipelineQuery, Response]
|
||||
extends PipelineConfig {
|
||||
|
||||
override val identifier: ProductPipelineIdentifier
|
||||
|
||||
val product: Product
|
||||
val paramConfig: ProductParamConfig
|
||||
|
||||
/**
|
||||
* Product Pipeline Gates will be executed before any other step (including retrieval from mixer
|
||||
* pipelines). They're executed sequentially, and any "Stop" result will prevent pipeline execution.
|
||||
*/
|
||||
def gates: Seq[Gate[Query]] = Seq.empty
|
||||
|
||||
def pipelineQueryTransformer(request: TRequest, params: Params): Query
|
||||
|
||||
/**
|
||||
* A list of all pipelines that power this product directly (there is no need to include pipelines
|
||||
* called by those pipelines).
|
||||
*
|
||||
* Only pipeline from this list should referenced from the pipelineSelector
|
||||
*/
|
||||
def pipelines: Seq[PipelineConfig]
|
||||
|
||||
/**
|
||||
* A pipeline selector selects a pipeline (from the list in `def pipelines`) to handle the
|
||||
* current request.
|
||||
*/
|
||||
def pipelineSelector(query: Query): ComponentIdentifier
|
||||
|
||||
/**
|
||||
** [[qualityFactorConfigs]] associates [[QualityFactorConfig]]s to specific pipelines
|
||||
* using [[ComponentIdentifier]].
|
||||
*/
|
||||
def qualityFactorConfigs: Map[ComponentIdentifier, QualityFactorConfig] =
|
||||
Map.empty
|
||||
|
||||
/**
|
||||
* By default (for safety), product mixer pipelines do not allow logged out requests.
|
||||
* A "DenyLoggedOutUsersGate" will be generated and added to the pipeline.
|
||||
*
|
||||
* You can disable this behavior by overriding `denyLoggedOutUsers` with False.
|
||||
*/
|
||||
val denyLoggedOutUsers: Boolean = true
|
||||
|
||||
/**
|
||||
* A pipeline can define a partial function to rescue failures here. They will be treated as failures
|
||||
* from a monitoring standpoint, and cancellation exceptions will always be propagated (they cannot be caught here).
|
||||
*/
|
||||
def failureClassifier: PartialFunction[Throwable, PipelineFailure] = PartialFunction.empty
|
||||
|
||||
/**
|
||||
* Alerts can be used to indicate the pipeline's service level objectives. Alerts and
|
||||
* dashboards will be automatically created based on this information.
|
||||
*/
|
||||
val alerts: Seq[Alert] = Seq.empty
|
||||
|
||||
/**
|
||||
* Access Policies can be used to gate who can query a product from Product Mixer's query tool
|
||||
* (go/turntable).
|
||||
*
|
||||
* This will typically be gated by an LDAP group associated with your team. For example:
|
||||
*
|
||||
* {{{
|
||||
* override val debugAccessPolicies: Set[AccessPolicy] = Set(AllowedLdapGroups("NAME"))
|
||||
* }}}
|
||||
*
|
||||
* You can disable all queries by using the [[com.twitter.product_mixer.core.functional_component.common.access_policy.BlockEverything]] policy.
|
||||
*/
|
||||
val debugAccessPolicies: Set[AccessPolicy]
|
||||
}
|
||||
|
||||
object ProductPipelineConfig extends PipelineConfigCompanion {
|
||||
val pipelineQueryTransformerStep: PipelineStepIdentifier = PipelineStepIdentifier(
|
||||
"PipelineQueryTransformer")
|
||||
val qualityFactorStep: PipelineStepIdentifier = PipelineStepIdentifier("QualityFactor")
|
||||
val gatesStep: PipelineStepIdentifier = PipelineStepIdentifier("Gates")
|
||||
val pipelineSelectorStep: PipelineStepIdentifier = PipelineStepIdentifier("PipelineSelector")
|
||||
val pipelineExecutionStep: PipelineStepIdentifier = PipelineStepIdentifier("PipelineExecution")
|
||||
|
||||
/** All the Steps which are executed by a [[ProductPipeline]] in the order in which they are run */
|
||||
override val stepsInOrder: Seq[PipelineStepIdentifier] = Seq(
|
||||
pipelineQueryTransformerStep,
|
||||
qualityFactorStep,
|
||||
gatesStep,
|
||||
pipelineSelectorStep,
|
||||
pipelineExecutionStep
|
||||
)
|
||||
}
|
Binary file not shown.
@ -1,5 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.product
|
||||
|
||||
import com.twitter.timelines.configapi.Params
|
||||
|
||||
case class ProductPipelineRequest[RequestType](request: RequestType, params: Params)
|
Binary file not shown.
@ -1,62 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.product
|
||||
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineResult
|
||||
import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineResult
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.product_mixer.core.pipeline.recommendation.RecommendationPipelineResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.pipeline_selector_executor.PipelineSelectorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.quality_factor_executor.QualityFactorExecutorResult
|
||||
|
||||
case class ProductPipelineResult[Result](
|
||||
transformedQuery: Option[PipelineQuery],
|
||||
qualityFactorResult: Option[QualityFactorExecutorResult],
|
||||
gateResult: Option[GateExecutorResult],
|
||||
pipelineSelectorResult: Option[PipelineSelectorExecutorResult],
|
||||
mixerPipelineResult: Option[MixerPipelineResult[Result]],
|
||||
recommendationPipelineResult: Option[RecommendationPipelineResult[_, Result]],
|
||||
traceId: Option[String],
|
||||
failure: Option[PipelineFailure],
|
||||
result: Option[Result])
|
||||
extends PipelineResult[Result] {
|
||||
|
||||
override val resultSize: Int = {
|
||||
if (mixerPipelineResult.isDefined) {
|
||||
mixerPipelineResult.map(_.resultSize).getOrElse(0)
|
||||
} else {
|
||||
recommendationPipelineResult.map(_.resultSize).getOrElse(0)
|
||||
}
|
||||
}
|
||||
|
||||
override def withFailure(failure: PipelineFailure): PipelineResult[Result] =
|
||||
copy(failure = Some(failure))
|
||||
|
||||
override def withResult(result: Result): PipelineResult[Result] = copy(result = Some(result))
|
||||
}
|
||||
|
||||
object ProductPipelineResult {
|
||||
def empty[A]: ProductPipelineResult[A] = ProductPipelineResult(
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None
|
||||
)
|
||||
|
||||
def fromResult[A](result: A): ProductPipelineResult[A] = ProductPipelineResult(
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
Some(result)
|
||||
)
|
||||
}
|
@ -1,62 +0,0 @@
|
||||
scala_library(
|
||||
sources = ["*.scala"],
|
||||
compiler_option_sets = ["fatal_warnings"],
|
||||
platform = "java8",
|
||||
strict_deps = True,
|
||||
tags = ["bazel-compatible"],
|
||||
dependencies = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector:insert_append_results",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featurestorev1",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/premarshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/selector",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/side_effect",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/async_feature_map_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_pipeline_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/filter_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/quality_factor_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/scoring_pipeline_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/selector_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util",
|
||||
"stitch/stitch-core",
|
||||
],
|
||||
exports = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/premarshaller",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/selector",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/side_effect",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/async_feature_map_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_pipeline_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/filter_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/quality_factor_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/scoring_pipeline_executor",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/selector_executor",
|
||||
"stitch/stitch-core",
|
||||
],
|
||||
)
|
Binary file not shown.
Binary file not shown.
@ -1,29 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.recommendation
|
||||
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.RecommendationPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.pipeline.Pipeline
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.stitch.Arrow
|
||||
|
||||
/**
|
||||
* A Recommendation Pipeline
|
||||
*
|
||||
* This is an abstract class, as we only construct these via the [[RecommendationPipelineBuilder]].
|
||||
*
|
||||
* A [[RecommendationPipeline]] is capable of processing requests (queries) and returning responses (results)
|
||||
* in the correct format to directly send to users.
|
||||
*
|
||||
* @tparam Query the domain model for the query or request
|
||||
* @tparam Candidate the type of the candidates
|
||||
* @tparam Result the final marshalled result type
|
||||
*/
|
||||
abstract class RecommendationPipeline[
|
||||
Query <: PipelineQuery,
|
||||
Candidate <: UniversalNoun[Any],
|
||||
Result]
|
||||
extends Pipeline[Query, Result] {
|
||||
override private[core] val config: RecommendationPipelineConfig[Query, Candidate, _, Result]
|
||||
override val arrow: Arrow[Query, RecommendationPipelineResult[Candidate, Result]]
|
||||
override val identifier: RecommendationPipelineIdentifier
|
||||
}
|
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
@ -1,67 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.recommendation
|
||||
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.marshalling.HasMarshalling
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineBuilderFactory
|
||||
import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipelineBuilderFactory
|
||||
import com.twitter.product_mixer.core.service.candidate_decorator_executor.CandidateDecoratorExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_pipeline_executor.CandidatePipelineExecutor
|
||||
import com.twitter.product_mixer.core.service.domain_marshaller_executor.DomainMarshallerExecutor
|
||||
import com.twitter.product_mixer.core.service.filter_executor.FilterExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.pipeline_result_side_effect_executor.PipelineResultSideEffectExecutor
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutor
|
||||
import com.twitter.product_mixer.core.service.query_feature_hydrator_executor.QueryFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.scoring_pipeline_executor.ScoringPipelineExecutor
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutor
|
||||
import com.twitter.product_mixer.core.service.transport_marshaller_executor.TransportMarshallerExecutor
|
||||
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
|
||||
@Singleton
|
||||
class RecommendationPipelineBuilderFactory @Inject() (
|
||||
candidatePipelineExecutor: CandidatePipelineExecutor,
|
||||
gateExecutor: GateExecutor,
|
||||
selectorExecutor: SelectorExecutor,
|
||||
queryFeatureHydratorExecutor: QueryFeatureHydratorExecutor,
|
||||
asyncFeatureMapExecutor: AsyncFeatureMapExecutor,
|
||||
candidateFeatureHydratorExecutor: CandidateFeatureHydratorExecutor,
|
||||
filterExecutor: FilterExecutor,
|
||||
scoringPipelineExecutor: ScoringPipelineExecutor,
|
||||
candidateDecoratorExecutor: CandidateDecoratorExecutor,
|
||||
domainMarshallerExecutor: DomainMarshallerExecutor,
|
||||
transportMarshallerExecutor: TransportMarshallerExecutor,
|
||||
pipelineResultSideEffectExecutor: PipelineResultSideEffectExecutor,
|
||||
candidatePipelineBuilderFactory: CandidatePipelineBuilderFactory,
|
||||
scoringPipelineBuilderFactory: ScoringPipelineBuilderFactory,
|
||||
statsReceiver: StatsReceiver) {
|
||||
|
||||
def get[
|
||||
Query <: PipelineQuery,
|
||||
Candidate <: UniversalNoun[Any],
|
||||
DomainResultType <: HasMarshalling,
|
||||
Result
|
||||
]: RecommendationPipelineBuilder[Query, Candidate, DomainResultType, Result] = {
|
||||
new RecommendationPipelineBuilder[Query, Candidate, DomainResultType, Result](
|
||||
candidatePipelineExecutor,
|
||||
gateExecutor,
|
||||
selectorExecutor,
|
||||
queryFeatureHydratorExecutor,
|
||||
asyncFeatureMapExecutor,
|
||||
candidateFeatureHydratorExecutor,
|
||||
filterExecutor,
|
||||
scoringPipelineExecutor,
|
||||
candidateDecoratorExecutor,
|
||||
domainMarshallerExecutor,
|
||||
transportMarshallerExecutor,
|
||||
pipelineResultSideEffectExecutor,
|
||||
candidatePipelineBuilderFactory,
|
||||
scoringPipelineBuilderFactory,
|
||||
statsReceiver
|
||||
)
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,262 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.recommendation
|
||||
|
||||
import com.twitter.product_mixer.component_library.selector.InsertAppendResults
|
||||
import com.twitter.product_mixer.core.functional_component.common.AllPipelines
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator
|
||||
import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator
|
||||
import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseQueryFeatureHydrator
|
||||
import com.twitter.product_mixer.core.functional_component.filter.Filter
|
||||
import com.twitter.product_mixer.core.functional_component.gate.Gate
|
||||
import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller
|
||||
import com.twitter.product_mixer.core.functional_component.selector.Selector
|
||||
import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect
|
||||
import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.RecommendationPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.model.marshalling.HasMarshalling
|
||||
import com.twitter.product_mixer.core.pipeline.FailOpenPolicy
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfigCompanion
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.ClosedGate
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipelineConfig
|
||||
import com.twitter.product_mixer.core.quality_factor.QualityFactorConfig
|
||||
|
||||
/**
|
||||
* This is the configuration necessary to generate a Recommendation Pipeline. Product code should create a
|
||||
* RecommendationPipelineConfig, and then use a RecommendationPipelineBuilder to get the final RecommendationPipeline which can
|
||||
* process requests.
|
||||
*
|
||||
* @tparam Query - The domain model for the query or request
|
||||
* @tparam Candidate - The type of the candidates that the Candidate Pipelines are generating
|
||||
* @tparam UnmarshalledResultType - The result type of the pipeline, but before marshalling to a wire protocol like URT
|
||||
* @tparam Result - The final result that will be served to users
|
||||
*/
|
||||
trait RecommendationPipelineConfig[
|
||||
Query <: PipelineQuery,
|
||||
Candidate <: UniversalNoun[Any],
|
||||
UnmarshalledResultType <: HasMarshalling,
|
||||
Result]
|
||||
extends PipelineConfig {
|
||||
|
||||
override val identifier: RecommendationPipelineIdentifier
|
||||
|
||||
/**
|
||||
* Recommendation Pipeline Gates will be executed before any other step (including retrieval from candidate
|
||||
* pipelines). They're executed sequentially, and any "Stop" result will prevent pipeline execution.
|
||||
*/
|
||||
def gates: Seq[Gate[Query]] = Seq.empty
|
||||
|
||||
/**
|
||||
* A recommendation pipeline can fetch query-level features before candidate pipelines are executed.
|
||||
*/
|
||||
def fetchQueryFeatures: Seq[BaseQueryFeatureHydrator[Query, _]] = Seq.empty
|
||||
|
||||
/**
|
||||
* Candidate pipelines retrieve candidates for possible inclusion in the result
|
||||
*/
|
||||
def fetchQueryFeaturesPhase2: Seq[BaseQueryFeatureHydrator[Query, _]] = Seq.empty
|
||||
|
||||
/**
|
||||
* What candidate pipelines should this Recommendations Pipeline get candidate from?
|
||||
*/
|
||||
def candidatePipelines: Seq[CandidatePipelineConfig[Query, _, _, _]]
|
||||
|
||||
/**
|
||||
* Dependent candidate pipelines to retrieve candidates that depend on the result of [[candidatePipelines]]
|
||||
* [[DependentCandidatePipelineConfig]] have access to the list of previously retrieved & decorated
|
||||
* candidates for use in constructing the query object.
|
||||
*/
|
||||
def dependentCandidatePipelines: Seq[DependentCandidatePipelineConfig[Query, _, _, _]] = Seq.empty
|
||||
|
||||
/**
|
||||
* Takes final ranked list of candidates & apply any business logic (e.g, deduplicating and merging
|
||||
* candidates before scoring).
|
||||
*/
|
||||
def postCandidatePipelinesSelectors: Seq[Selector[Query]] = Seq(InsertAppendResults(AllPipelines))
|
||||
|
||||
/**
|
||||
* After selectors are run, you can fetch features for each candidate.
|
||||
* The existing features from previous hydrations are passed in as inputs. You are not expected to
|
||||
* put them into the resulting feature map yourself - they will be merged for you by the platform.
|
||||
*/
|
||||
def postCandidatePipelinesFeatureHydration: Seq[
|
||||
BaseCandidateFeatureHydrator[Query, Candidate, _]
|
||||
] =
|
||||
Seq.empty
|
||||
|
||||
/**
|
||||
* Global filters to run on all candidates.
|
||||
*/
|
||||
def globalFilters: Seq[Filter[Query, Candidate]] = Seq.empty
|
||||
|
||||
/**
|
||||
* By default, a Recommendation Pipeline will fail closed - if any candidate or scoring
|
||||
* pipeline fails to return a result, then the Recommendation Pipeline will not return a result.
|
||||
* You can adjust this default policy, or provide specific policies to specific pipelines.
|
||||
* Those specific policies will take priority.
|
||||
*
|
||||
* FailOpenPolicy.All will always fail open (the RecommendationPipeline will continue without that pipeline)
|
||||
* FailOpenPolicy.Never will always fail closed (the RecommendationPipeline will fail if that pipeline fails)
|
||||
*
|
||||
* There's a default policy, and a specific Map of policies that takes precedence.
|
||||
*/
|
||||
def defaultFailOpenPolicy: FailOpenPolicy = FailOpenPolicy(Set(ClosedGate))
|
||||
def candidatePipelineFailOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy] =
|
||||
Map.empty
|
||||
def scoringPipelineFailOpenPolicies: Map[ScoringPipelineIdentifier, FailOpenPolicy] = Map.empty
|
||||
|
||||
/**
|
||||
** [[qualityFactorConfigs]] associates [[QualityFactorConfig]]s to specific candidate pipelines
|
||||
* using [[ComponentIdentifier]].
|
||||
*/
|
||||
def qualityFactorConfigs: Map[ComponentIdentifier, QualityFactorConfig] =
|
||||
Map.empty
|
||||
|
||||
/**
|
||||
* Scoring pipelines for scoring candidates.
|
||||
* @note These do not drop or re-order candidates, you should do those in the sub-sequent selectors
|
||||
* step based off of the scores on candidates set in those [[ScoringPipeline]]s.
|
||||
*/
|
||||
def scoringPipelines: Seq[ScoringPipelineConfig[Query, Candidate]]
|
||||
|
||||
/**
|
||||
* Takes final ranked list of candidates & apply any business logic (e.g, capping number
|
||||
* of ad accounts or pacing ad accounts).
|
||||
*/
|
||||
def resultSelectors: Seq[Selector[Query]]
|
||||
|
||||
/**
|
||||
* Takes the final selected list of candidates and applies a final list of filters.
|
||||
* Useful for doing very expensive filtering at the end of your pipeline.
|
||||
*/
|
||||
def postSelectionFilters: Seq[Filter[Query, Candidate]] = Seq.empty
|
||||
|
||||
/**
|
||||
* Decorators allow for adding Presentations to candidates. While the Presentation can contain any
|
||||
* arbitrary data, Decorators are often used to add a UrtItemPresentation for URT item support. Most
|
||||
* customers will prefer to set a decorator in their respective candidate pipeline, however, a final
|
||||
* global one is available for those that do global decoration as late possible to avoid unnecessary hydrations.
|
||||
* @note This decorator can only return an ItemPresentation.
|
||||
* @note This decorator cannot decorate an already decorated candidate from the prior decorator
|
||||
* step in candidate pipelines.
|
||||
*/
|
||||
def decorator: Option[CandidateDecorator[Query, Candidate]] = None
|
||||
|
||||
/**
|
||||
* Domain marshaller transforms the selections into the model expected by the marshaller
|
||||
*/
|
||||
def domainMarshaller: DomainMarshaller[Query, UnmarshalledResultType]
|
||||
|
||||
/**
|
||||
* Mixer result side effects that are executed after selection and domain marshalling
|
||||
*/
|
||||
def resultSideEffects: Seq[PipelineResultSideEffect[Query, UnmarshalledResultType]] = Seq()
|
||||
|
||||
/**
|
||||
* Transport marshaller transforms the model into our line-level API like URT or JSON
|
||||
*/
|
||||
def transportMarshaller: TransportMarshaller[UnmarshalledResultType, Result]
|
||||
|
||||
/**
|
||||
* A pipeline can define a partial function to rescue failures here. They will be treated as failures
|
||||
* from a monitoring standpoint, and cancellation exceptions will always be propagated (they cannot be caught here).
|
||||
*/
|
||||
def failureClassifier: PartialFunction[Throwable, PipelineFailure] = PartialFunction.empty
|
||||
|
||||
/**
|
||||
* Alerts can be used to indicate the pipeline's service level objectives. Alerts and
|
||||
* dashboards will be automatically created based on this information.
|
||||
*/
|
||||
val alerts: Seq[Alert] = Seq.empty
|
||||
|
||||
/**
|
||||
* This method is used by the product mixer framework to build the pipeline.
|
||||
*/
|
||||
private[core] final def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
builder: RecommendationPipelineBuilderFactory
|
||||
): RecommendationPipeline[Query, Candidate, Result] =
|
||||
builder.get.build(parentComponentIdentifierStack, this)
|
||||
}
|
||||
|
||||
object RecommendationPipelineConfig extends PipelineConfigCompanion {
|
||||
val qualityFactorStep: PipelineStepIdentifier = PipelineStepIdentifier("QualityFactor")
|
||||
val gatesStep: PipelineStepIdentifier = PipelineStepIdentifier("Gates")
|
||||
val fetchQueryFeaturesStep: PipelineStepIdentifier = PipelineStepIdentifier("FetchQueryFeatures")
|
||||
val fetchQueryFeaturesPhase2Step: PipelineStepIdentifier = PipelineStepIdentifier(
|
||||
"FetchQueryFeaturesPhase2")
|
||||
val candidatePipelinesStep: PipelineStepIdentifier = PipelineStepIdentifier("CandidatePipelines")
|
||||
val dependentCandidatePipelinesStep: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("DependentCandidatePipelines")
|
||||
val postCandidatePipelinesSelectorsStep: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("PostCandidatePipelinesSelectors")
|
||||
val postCandidatePipelinesFeatureHydrationStep: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("PostCandidatePipelinesFeatureHydration")
|
||||
val globalFiltersStep: PipelineStepIdentifier = PipelineStepIdentifier("GlobalFilters")
|
||||
val scoringPipelinesStep: PipelineStepIdentifier = PipelineStepIdentifier("ScoringPipelines")
|
||||
val resultSelectorsStep: PipelineStepIdentifier = PipelineStepIdentifier("ResultSelectors")
|
||||
val postSelectionFiltersStep: PipelineStepIdentifier = PipelineStepIdentifier(
|
||||
"PostSelectionFilters")
|
||||
val decoratorStep: PipelineStepIdentifier = PipelineStepIdentifier("Decorator")
|
||||
val domainMarshallerStep: PipelineStepIdentifier = PipelineStepIdentifier("DomainMarshaller")
|
||||
val resultSideEffectsStep: PipelineStepIdentifier = PipelineStepIdentifier("ResultSideEffects")
|
||||
val transportMarshallerStep: PipelineStepIdentifier = PipelineStepIdentifier(
|
||||
"TransportMarshaller")
|
||||
|
||||
/** All the Steps which are executed by a [[RecommendationPipeline]] in the order in which they are run */
|
||||
override val stepsInOrder: Seq[PipelineStepIdentifier] = Seq(
|
||||
qualityFactorStep,
|
||||
gatesStep,
|
||||
fetchQueryFeaturesStep,
|
||||
fetchQueryFeaturesPhase2Step,
|
||||
asyncFeaturesStep(candidatePipelinesStep),
|
||||
candidatePipelinesStep,
|
||||
asyncFeaturesStep(dependentCandidatePipelinesStep),
|
||||
dependentCandidatePipelinesStep,
|
||||
asyncFeaturesStep(postCandidatePipelinesSelectorsStep),
|
||||
postCandidatePipelinesSelectorsStep,
|
||||
asyncFeaturesStep(postCandidatePipelinesFeatureHydrationStep),
|
||||
postCandidatePipelinesFeatureHydrationStep,
|
||||
asyncFeaturesStep(globalFiltersStep),
|
||||
globalFiltersStep,
|
||||
asyncFeaturesStep(scoringPipelinesStep),
|
||||
scoringPipelinesStep,
|
||||
asyncFeaturesStep(resultSelectorsStep),
|
||||
resultSelectorsStep,
|
||||
asyncFeaturesStep(postSelectionFiltersStep),
|
||||
postSelectionFiltersStep,
|
||||
asyncFeaturesStep(decoratorStep),
|
||||
decoratorStep,
|
||||
domainMarshallerStep,
|
||||
asyncFeaturesStep(resultSideEffectsStep),
|
||||
resultSideEffectsStep,
|
||||
transportMarshallerStep
|
||||
)
|
||||
|
||||
/**
|
||||
* All the Steps which an [[com.twitter.product_mixer.core.functional_component.feature_hydrator.AsyncHydrator AsyncHydrator]]
|
||||
* can be configured to [[com.twitter.product_mixer.core.functional_component.feature_hydrator.AsyncHydrator.hydrateBefore hydrateBefore]]
|
||||
*/
|
||||
override val stepsAsyncFeatureHydrationCanBeCompletedBy: Set[PipelineStepIdentifier] = Set(
|
||||
candidatePipelinesStep,
|
||||
dependentCandidatePipelinesStep,
|
||||
postCandidatePipelinesSelectorsStep,
|
||||
postCandidatePipelinesFeatureHydrationStep,
|
||||
globalFiltersStep,
|
||||
scoringPipelinesStep,
|
||||
resultSelectorsStep,
|
||||
postSelectionFiltersStep,
|
||||
decoratorStep,
|
||||
resultSideEffectsStep,
|
||||
)
|
||||
}
|
Binary file not shown.
@ -1,84 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.recommendation
|
||||
|
||||
import com.twitter.product_mixer.core.feature.featuremap.asyncfeaturemap.AsyncFeatureMap
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.marshalling.HasMarshalling
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineResult
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.product_mixer.core.service.async_feature_map_executor.AsyncFeatureMapExecutorResults
|
||||
import com.twitter.product_mixer.core.service.candidate_decorator_executor.CandidateDecoratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.candidate_pipeline_executor.CandidatePipelineExecutorResult
|
||||
import com.twitter.product_mixer.core.service.domain_marshaller_executor.DomainMarshallerExecutor
|
||||
import com.twitter.product_mixer.core.service.filter_executor.FilterExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.pipeline_result_side_effect_executor.PipelineResultSideEffectExecutor
|
||||
import com.twitter.product_mixer.core.service.quality_factor_executor.QualityFactorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.query_feature_hydrator_executor.QueryFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.scoring_pipeline_executor.ScoringPipelineExecutorResult
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.transport_marshaller_executor.TransportMarshallerExecutor
|
||||
|
||||
case class RecommendationPipelineResult[Candidate <: UniversalNoun[Any], ResultType](
|
||||
qualityFactorResult: Option[QualityFactorExecutorResult],
|
||||
gateResult: Option[GateExecutorResult],
|
||||
queryFeatures: Option[QueryFeatureHydratorExecutor.Result],
|
||||
queryFeaturesPhase2: Option[QueryFeatureHydratorExecutor.Result],
|
||||
mergedAsyncQueryFeatures: Option[AsyncFeatureMap],
|
||||
candidatePipelineResults: Option[CandidatePipelineExecutorResult],
|
||||
dependentCandidatePipelineResults: Option[CandidatePipelineExecutorResult],
|
||||
postCandidatePipelinesSelectorResults: Option[SelectorExecutorResult],
|
||||
postCandidatePipelinesFeatureHydrationResults: Option[
|
||||
CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
],
|
||||
globalFilterResults: Option[FilterExecutorResult[Candidate]],
|
||||
scoringPipelineResults: Option[ScoringPipelineExecutorResult[Candidate]],
|
||||
resultSelectorResults: Option[SelectorExecutorResult],
|
||||
postSelectionFilterResults: Option[FilterExecutorResult[Candidate]],
|
||||
candidateDecoratorResult: Option[CandidateDecoratorExecutorResult],
|
||||
domainMarshallerResults: Option[DomainMarshallerExecutor.Result[HasMarshalling]],
|
||||
resultSideEffectResults: Option[PipelineResultSideEffectExecutor.Result],
|
||||
asyncFeatureHydrationResults: Option[AsyncFeatureMapExecutorResults],
|
||||
transportMarshallerResults: Option[TransportMarshallerExecutor.Result[ResultType]],
|
||||
failure: Option[PipelineFailure],
|
||||
result: Option[ResultType])
|
||||
extends PipelineResult[ResultType] {
|
||||
override val resultSize: Int = result match {
|
||||
case Some(seqResult @ Seq(_)) => seqResult.length
|
||||
case Some(_) => 1
|
||||
case None => 0
|
||||
}
|
||||
|
||||
override def withFailure(
|
||||
failure: PipelineFailure
|
||||
): RecommendationPipelineResult[Candidate, ResultType] =
|
||||
copy(failure = Some(failure))
|
||||
override def withResult(result: ResultType): RecommendationPipelineResult[Candidate, ResultType] =
|
||||
copy(result = Some(result))
|
||||
}
|
||||
|
||||
object RecommendationPipelineResult {
|
||||
def empty[A <: UniversalNoun[Any], B]: RecommendationPipelineResult[A, B] =
|
||||
RecommendationPipelineResult(
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None
|
||||
)
|
||||
}
|
@ -1,36 +0,0 @@
|
||||
scala_library(
|
||||
sources = ["*.scala"],
|
||||
compiler_option_sets = ["fatal_warnings"],
|
||||
platform = "java8",
|
||||
strict_deps = True,
|
||||
tags = ["bazel-compatible"],
|
||||
dependencies = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate",
|
||||
"product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector:insert_append_results",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/candidate_feature_hydrator",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/scorer",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/selector",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util",
|
||||
"stitch/stitch-core",
|
||||
"util/util-core:scala",
|
||||
],
|
||||
exports = [
|
||||
"3rdparty/jvm/javax/inject:javax.inject",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/candidate_feature_hydrator",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/gate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/scorer",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/selector",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice",
|
||||
"stitch/stitch-core",
|
||||
],
|
||||
)
|
Binary file not shown.
Binary file not shown.
@ -1,202 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.scoring
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.decorator.Decoration
|
||||
import com.twitter.product_mixer.core.functional_component.scorer.ScoredCandidateResult
|
||||
import com.twitter.product_mixer.core.gate.ParamGate
|
||||
import com.twitter.product_mixer.core.gate.ParamGate._
|
||||
import com.twitter.product_mixer.core.model.common.CandidateWithFeatures
|
||||
import com.twitter.product_mixer.core.model.common.Component
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails
|
||||
import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails
|
||||
import com.twitter.product_mixer.core.pipeline.NewPipelineBuilder
|
||||
import com.twitter.product_mixer.core.pipeline.NewPipelineArrowBuilder
|
||||
import com.twitter.product_mixer.core.pipeline.NewPipelineResult
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.ClosedGate
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailureClassifier
|
||||
import com.twitter.product_mixer.core.pipeline.state.HasCandidatesWithDetails
|
||||
import com.twitter.product_mixer.core.pipeline.state.HasCandidatesWithFeatures
|
||||
import com.twitter.product_mixer.core.pipeline.state.HasExecutorResults
|
||||
import com.twitter.product_mixer.core.pipeline.state.HasQuery
|
||||
import com.twitter.product_mixer.core.pipeline.state.HasResult
|
||||
import com.twitter.product_mixer.core.pipeline.step.candidate_feature_hydrator.CandidateFeatureHydratorStep
|
||||
import com.twitter.product_mixer.core.pipeline.step.gate.GateStep
|
||||
import com.twitter.product_mixer.core.pipeline.step.scorer.ScorerStep
|
||||
import com.twitter.product_mixer.core.pipeline.step.selector.SelectorStep
|
||||
import com.twitter.product_mixer.core.service.Executor
|
||||
import com.twitter.product_mixer.core.service.ExecutorResult
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.StoppedGateException
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutorResult
|
||||
import com.twitter.stitch.Arrow
|
||||
import javax.inject.Inject
|
||||
import scala.collection.immutable.ListMap
|
||||
|
||||
/**
|
||||
* NewScoringPipelineBuilder builds [[ScoringPipeline]]s from [[ScoringPipelineConfig]]s.
|
||||
* New because it's meant to eventually replace [[ScoringPipelineBuilder]]
|
||||
* You should inject a [[ScoringPipelineBuilderFactory]] and call `.get` to build these.
|
||||
*
|
||||
* @see [[ScoringPipelineConfig]] for the description of the type parameters
|
||||
* @tparam Query the type of query these accept.
|
||||
* @tparam Candidate the domain model for the candidate being scored
|
||||
*/
|
||||
class NewScoringPipelineBuilder[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]] @Inject() (
|
||||
selectionStep: SelectorStep[Query, ScoringPipelineState[Query, Candidate]],
|
||||
gateStep: GateStep[Query, ScoringPipelineState[Query, Candidate]],
|
||||
candidateFeatureHydrationStep: CandidateFeatureHydratorStep[
|
||||
Query,
|
||||
Candidate,
|
||||
ScoringPipelineState[Query, Candidate]
|
||||
],
|
||||
scorerStep: ScorerStep[Query, Candidate, ScoringPipelineState[Query, Candidate]])
|
||||
extends NewPipelineBuilder[ScoringPipelineConfig[Query, Candidate], Seq[
|
||||
CandidateWithFeatures[Candidate]
|
||||
], ScoringPipelineState[Query, Candidate], ScoringPipeline[Query, Candidate]] {
|
||||
|
||||
override def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
arrowBuilder: NewPipelineArrowBuilder[ArrowResult, ArrowState],
|
||||
scoringPipelineConfig: ScoringPipelineConfig[Query, Candidate]
|
||||
): ScoringPipeline[Query, Candidate] = {
|
||||
val pipelineIdentifier = scoringPipelineConfig.identifier
|
||||
|
||||
val context = Executor.Context(
|
||||
PipelineFailureClassifier(
|
||||
scoringPipelineConfig.failureClassifier.orElse(
|
||||
StoppedGateException.classifier(ClosedGate))),
|
||||
parentComponentIdentifierStack.push(pipelineIdentifier)
|
||||
)
|
||||
|
||||
val enabledGateOpt = scoringPipelineConfig.enabledDeciderParam.map { deciderParam =>
|
||||
ParamGate(pipelineIdentifier + EnabledGateSuffix, deciderParam)
|
||||
}
|
||||
val supportedClientGateOpt = scoringPipelineConfig.supportedClientParam.map { param =>
|
||||
ParamGate(pipelineIdentifier + SupportedClientGateSuffix, param)
|
||||
}
|
||||
|
||||
/**
|
||||
* Evaluate enabled decider gate first since if it's off, there is no reason to proceed
|
||||
* Next evaluate supported client feature switch gate, followed by customer configured gates
|
||||
*/
|
||||
val allGates =
|
||||
enabledGateOpt.toSeq ++ supportedClientGateOpt.toSeq ++ scoringPipelineConfig.gates
|
||||
|
||||
val underlyingArrow = arrowBuilder
|
||||
.add(ScoringPipelineConfig.gatesStep, gateStep, allGates)
|
||||
.add(ScoringPipelineConfig.selectorsStep, selectionStep, scoringPipelineConfig.selectors)
|
||||
.add(
|
||||
ScoringPipelineConfig.preScoringFeatureHydrationPhase1Step,
|
||||
candidateFeatureHydrationStep,
|
||||
scoringPipelineConfig.preScoringFeatureHydrationPhase1)
|
||||
.add(
|
||||
ScoringPipelineConfig.preScoringFeatureHydrationPhase2Step,
|
||||
candidateFeatureHydrationStep,
|
||||
scoringPipelineConfig.preScoringFeatureHydrationPhase2)
|
||||
.add(ScoringPipelineConfig.scorersStep, scorerStep, scoringPipelineConfig.scorers).buildArrow(
|
||||
context)
|
||||
|
||||
val finalArrow = Arrow
|
||||
.map { inputs: ScoringPipeline.Inputs[Query] =>
|
||||
ScoringPipelineState[Query, Candidate](inputs.query, inputs.candidates, ListMap.empty)
|
||||
}.andThen(underlyingArrow).map { pipelineResult =>
|
||||
ScoringPipelineResult(
|
||||
gateResults = pipelineResult.executorResultsByPipelineStep
|
||||
.get(ScoringPipelineConfig.gatesStep)
|
||||
.map(_.asInstanceOf[GateExecutorResult]),
|
||||
selectorResults = pipelineResult.executorResultsByPipelineStep
|
||||
.get(ScoringPipelineConfig.selectorsStep)
|
||||
.map(_.asInstanceOf[SelectorExecutorResult]),
|
||||
preScoringHydrationPhase1Result = pipelineResult.executorResultsByPipelineStep
|
||||
.get(ScoringPipelineConfig.preScoringFeatureHydrationPhase1Step)
|
||||
.map(_.asInstanceOf[CandidateFeatureHydratorExecutorResult[Candidate]]),
|
||||
preScoringHydrationPhase2Result = pipelineResult.executorResultsByPipelineStep
|
||||
.get(ScoringPipelineConfig.preScoringFeatureHydrationPhase2Step)
|
||||
.map(_.asInstanceOf[CandidateFeatureHydratorExecutorResult[Candidate]]),
|
||||
scorerResults = pipelineResult.executorResultsByPipelineStep
|
||||
.get(ScoringPipelineConfig.scorersStep)
|
||||
.map(_.asInstanceOf[CandidateFeatureHydratorExecutorResult[Candidate]]),
|
||||
failure = pipelineResult match {
|
||||
case failure: NewPipelineResult.Failure =>
|
||||
Some(failure.failure)
|
||||
case _ => None
|
||||
},
|
||||
result = pipelineResult match {
|
||||
case result: NewPipelineResult.Success[Seq[CandidateWithFeatures[Candidate]]] =>
|
||||
Some(result.result.map { candidateWithFeatures =>
|
||||
ScoredCandidateResult(
|
||||
candidateWithFeatures.candidate,
|
||||
candidateWithFeatures.features)
|
||||
})
|
||||
case _ => None
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
new ScoringPipeline[Query, Candidate] {
|
||||
override val arrow: Arrow[ScoringPipeline.Inputs[Query], ScoringPipelineResult[Candidate]] =
|
||||
finalArrow
|
||||
|
||||
override val identifier: ScoringPipelineIdentifier = scoringPipelineConfig.identifier
|
||||
|
||||
override val alerts: Seq[Alert] = scoringPipelineConfig.alerts
|
||||
|
||||
override val children: Seq[Component] =
|
||||
allGates ++ scoringPipelineConfig.preScoringFeatureHydrationPhase1 ++ scoringPipelineConfig.preScoringFeatureHydrationPhase2 ++ scoringPipelineConfig.scorers
|
||||
|
||||
override private[core] val config = scoringPipelineConfig
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
case class ScoringPipelineState[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]](
|
||||
override val query: Query,
|
||||
candidates: Seq[ItemCandidateWithDetails],
|
||||
override val executorResultsByPipelineStep: ListMap[PipelineStepIdentifier, ExecutorResult])
|
||||
extends HasQuery[Query, ScoringPipelineState[Query, Candidate]]
|
||||
with HasCandidatesWithDetails[ScoringPipelineState[Query, Candidate]]
|
||||
with HasCandidatesWithFeatures[Candidate, ScoringPipelineState[Query, Candidate]]
|
||||
with HasExecutorResults[ScoringPipelineState[Query, Candidate]]
|
||||
with HasResult[Seq[CandidateWithFeatures[Candidate]]] {
|
||||
|
||||
override val candidatesWithDetails: Seq[CandidateWithDetails] = candidates
|
||||
|
||||
override val candidatesWithFeatures: Seq[CandidateWithFeatures[Candidate]] =
|
||||
candidates.asInstanceOf[Seq[CandidateWithFeatures[Candidate]]]
|
||||
|
||||
override val buildResult: Seq[CandidateWithFeatures[Candidate]] = candidatesWithFeatures
|
||||
|
||||
override def updateCandidatesWithDetails(
|
||||
newCandidates: Seq[CandidateWithDetails]
|
||||
): ScoringPipelineState[Query, Candidate] = {
|
||||
this.copy(candidates = newCandidates.asInstanceOf[Seq[ItemCandidateWithDetails]])
|
||||
}
|
||||
|
||||
override def updateQuery(newQuery: Query): ScoringPipelineState[Query, Candidate] =
|
||||
this.copy(query = newQuery)
|
||||
|
||||
override def updateDecorations(
|
||||
decoration: Seq[Decoration]
|
||||
): ScoringPipelineState[Query, Candidate] = ???
|
||||
|
||||
override def updateCandidatesWithFeatures(
|
||||
newCandidates: Seq[CandidateWithFeatures[Candidate]]
|
||||
): ScoringPipelineState[Query, Candidate] = {
|
||||
val updatedCandidates = candidates.zip(newCandidates).map {
|
||||
case (itemCandidateWithDetails, newCandidate) =>
|
||||
itemCandidateWithDetails.copy(features =
|
||||
itemCandidateWithDetails.features ++ newCandidate.features)
|
||||
}
|
||||
this.copy(query, updatedCandidates)
|
||||
}
|
||||
|
||||
override private[pipeline] def setExecutorResults(
|
||||
newMap: ListMap[PipelineStepIdentifier, ExecutorResult]
|
||||
) = this.copy(executorResultsByPipelineStep = newMap)
|
||||
}
|
Binary file not shown.
@ -1,32 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.scoring
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.scorer.ScoredCandidateResult
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails
|
||||
import com.twitter.product_mixer.core.pipeline.Pipeline
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.stitch.Arrow
|
||||
|
||||
/**
|
||||
* A Scoring Pipeline
|
||||
*
|
||||
* This is an abstract class, as we only construct these via the [[ScoringPipelineBuilder]].
|
||||
*
|
||||
* A [[ScoringPipeline]] is capable of pre-filtering candidates for scoring, performing the scoring
|
||||
* then running selection heuristics (ranking, dropping, etc) based off of the score.
|
||||
* @tparam Query the domain model for the query or request
|
||||
* @tparam Candidate the domain model for the candidate being scored
|
||||
*/
|
||||
abstract class ScoringPipeline[-Query <: PipelineQuery, Candidate <: UniversalNoun[Any]]
|
||||
extends Pipeline[ScoringPipeline.Inputs[Query], Seq[ScoredCandidateResult[Candidate]]] {
|
||||
override private[core] val config: ScoringPipelineConfig[Query, Candidate]
|
||||
override val arrow: Arrow[ScoringPipeline.Inputs[Query], ScoringPipelineResult[Candidate]]
|
||||
override val identifier: ScoringPipelineIdentifier
|
||||
}
|
||||
|
||||
object ScoringPipeline {
|
||||
case class Inputs[+Query <: PipelineQuery](
|
||||
query: Query,
|
||||
candidates: Seq[ItemCandidateWithDetails])
|
||||
}
|
Binary file not shown.
@ -1,367 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.scoring
|
||||
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.product_mixer.core.feature.featuremap.FeatureMap
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.scorer.ScoredCandidateResult
|
||||
import com.twitter.product_mixer.core.gate.ParamGate
|
||||
import com.twitter.product_mixer.core.gate.ParamGate.EnabledGateSuffix
|
||||
import com.twitter.product_mixer.core.gate.ParamGate.SupportedClientGateSuffix
|
||||
import com.twitter.product_mixer.core.model.common.CandidateWithFeatures
|
||||
import com.twitter.product_mixer.core.model.common.Component
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails
|
||||
import com.twitter.product_mixer.core.pipeline.InvalidStepStateException
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineBuilder
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.ClosedGate
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailureClassifier
|
||||
import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipeline.Inputs
|
||||
import com.twitter.product_mixer.core.service.Executor
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.StoppedGateException
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutor
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutorResult
|
||||
import com.twitter.stitch.Arrow
|
||||
import javax.inject.Inject
|
||||
|
||||
/**
|
||||
* ScoringPipelineBuilder builds [[ScoringPipeline]]s from [[ScoringPipelineConfig]]s.
|
||||
*
|
||||
* You should inject a [[ScoringPipelineBuilderFactory]] and call `.get` to build these.
|
||||
*
|
||||
* @see [[ScoringPipelineConfig]] for the description of the type parameters
|
||||
* @tparam Query the type of query these accept.
|
||||
* @tparam Candidate the domain model for the candidate being scored
|
||||
*/
|
||||
class ScoringPipelineBuilder[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]] @Inject() (
|
||||
gateExecutor: GateExecutor,
|
||||
selectorExecutor: SelectorExecutor,
|
||||
candidateFeatureHydratorExecutor: CandidateFeatureHydratorExecutor,
|
||||
override val statsReceiver: StatsReceiver)
|
||||
extends PipelineBuilder[Inputs[Query]] {
|
||||
|
||||
override type UnderlyingResultType = Seq[ScoredCandidateResult[Candidate]]
|
||||
override type PipelineResultType = ScoringPipelineResult[Candidate]
|
||||
|
||||
def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
config: ScoringPipelineConfig[Query, Candidate]
|
||||
): ScoringPipeline[Query, Candidate] = {
|
||||
|
||||
val pipelineIdentifier = config.identifier
|
||||
|
||||
val context = Executor.Context(
|
||||
PipelineFailureClassifier(
|
||||
config.failureClassifier.orElse(StoppedGateException.classifier(ClosedGate))),
|
||||
parentComponentIdentifierStack.push(pipelineIdentifier)
|
||||
)
|
||||
|
||||
val enabledGateOpt = config.enabledDeciderParam.map { deciderParam =>
|
||||
ParamGate(pipelineIdentifier + EnabledGateSuffix, deciderParam)
|
||||
}
|
||||
val supportedClientGateOpt = config.supportedClientParam.map { param =>
|
||||
ParamGate(pipelineIdentifier + SupportedClientGateSuffix, param)
|
||||
}
|
||||
|
||||
/**
|
||||
* Evaluate enabled decider gate first since if it's off, there is no reason to proceed
|
||||
* Next evaluate supported client feature switch gate, followed by customer configured gates
|
||||
*/
|
||||
val allGates = enabledGateOpt.toSeq ++ supportedClientGateOpt.toSeq ++ config.gates
|
||||
|
||||
val GatesStep = new Step[Query, GateExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = ScoringPipelineConfig.gatesStep
|
||||
|
||||
override lazy val executorArrow: Arrow[Query, GateExecutorResult] =
|
||||
gateExecutor.arrow(allGates, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ScoringPipeline.Inputs[Query],
|
||||
previousResult: ScoringPipelineResult[Candidate]
|
||||
): Query = {
|
||||
query.query
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ScoringPipelineResult[Candidate],
|
||||
executorResult: GateExecutorResult
|
||||
): ScoringPipelineResult[Candidate] =
|
||||
previousPipelineResult.copy(gateResults = Some(executorResult))
|
||||
}
|
||||
|
||||
val SelectorsStep = new Step[SelectorExecutor.Inputs[Query], SelectorExecutorResult] {
|
||||
override def identifier: PipelineStepIdentifier = ScoringPipelineConfig.selectorsStep
|
||||
|
||||
override def executorArrow: Arrow[SelectorExecutor.Inputs[Query], SelectorExecutorResult] =
|
||||
selectorExecutor.arrow(config.selectors, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ScoringPipeline.Inputs[Query],
|
||||
previousResult: ScoringPipelineResult[Candidate]
|
||||
): SelectorExecutor.Inputs[Query] = SelectorExecutor.Inputs(query.query, query.candidates)
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ScoringPipelineResult[Candidate],
|
||||
executorResult: SelectorExecutorResult
|
||||
): ScoringPipelineResult[Candidate] =
|
||||
previousPipelineResult.copy(selectorResults = Some(executorResult))
|
||||
}
|
||||
|
||||
val PreScoringFeatureHydrationPhase1Step =
|
||||
new Step[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Candidate],
|
||||
CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
ScoringPipelineConfig.preScoringFeatureHydrationPhase1Step
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Candidate],
|
||||
CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
] =
|
||||
candidateFeatureHydratorExecutor.arrow(config.preScoringFeatureHydrationPhase1, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ScoringPipeline.Inputs[Query],
|
||||
previousResult: ScoringPipelineResult[Candidate]
|
||||
): CandidateFeatureHydratorExecutor.Inputs[Query, Candidate] = {
|
||||
val selectedCandidatesResult = previousResult.selectorResults.getOrElse {
|
||||
throw InvalidStepStateException(identifier, "SelectorResults")
|
||||
}.selectedCandidates
|
||||
|
||||
CandidateFeatureHydratorExecutor.Inputs(
|
||||
query.query,
|
||||
selectedCandidatesResult.asInstanceOf[Seq[CandidateWithFeatures[Candidate]]])
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ScoringPipelineResult[Candidate],
|
||||
executorResult: CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
): ScoringPipelineResult[Candidate] = previousPipelineResult.copy(
|
||||
preScoringHydrationPhase1Result = Some(executorResult)
|
||||
)
|
||||
}
|
||||
|
||||
val PreScoringFeatureHydrationPhase2Step =
|
||||
new Step[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Candidate],
|
||||
CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
] {
|
||||
override def identifier: PipelineStepIdentifier =
|
||||
ScoringPipelineConfig.preScoringFeatureHydrationPhase2Step
|
||||
|
||||
override def executorArrow: Arrow[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Candidate],
|
||||
CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
] =
|
||||
candidateFeatureHydratorExecutor.arrow(config.preScoringFeatureHydrationPhase2, context)
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ScoringPipeline.Inputs[Query],
|
||||
previousResult: ScoringPipelineResult[Candidate]
|
||||
): CandidateFeatureHydratorExecutor.Inputs[Query, Candidate] = {
|
||||
val preScoringHydrationPhase1FeatureMaps: Seq[FeatureMap] =
|
||||
previousResult.preScoringHydrationPhase1Result
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(identifier, "PreScoringHydrationPhase1Result"))
|
||||
.results.map(_.features)
|
||||
|
||||
val itemCandidates = previousResult.selectorResults
|
||||
.getOrElse(throw InvalidStepStateException(identifier, "SelectionResults"))
|
||||
.selectedCandidates.collect {
|
||||
case itemCandidate: ItemCandidateWithDetails => itemCandidate
|
||||
}
|
||||
// If there is no feature hydration (empty results), no need to attempt merging.
|
||||
val candidates = if (preScoringHydrationPhase1FeatureMaps.isEmpty) {
|
||||
itemCandidates
|
||||
} else {
|
||||
itemCandidates.zip(preScoringHydrationPhase1FeatureMaps).map {
|
||||
case (itemCandidate, featureMap) =>
|
||||
itemCandidate.copy(features = itemCandidate.features ++ featureMap)
|
||||
}
|
||||
}
|
||||
|
||||
CandidateFeatureHydratorExecutor.Inputs(
|
||||
query.query,
|
||||
candidates.asInstanceOf[Seq[CandidateWithFeatures[Candidate]]])
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ScoringPipelineResult[Candidate],
|
||||
executorResult: CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
): ScoringPipelineResult[Candidate] = previousPipelineResult.copy(
|
||||
preScoringHydrationPhase2Result = Some(executorResult)
|
||||
)
|
||||
}
|
||||
|
||||
def getMergedPreScoringFeatureMap(
|
||||
stepIdentifier: PipelineStepIdentifier,
|
||||
previousResult: ScoringPipelineResult[Candidate]
|
||||
): Seq[FeatureMap] = {
|
||||
val preScoringHydrationPhase1FeatureMaps: Seq[FeatureMap] =
|
||||
previousResult.preScoringHydrationPhase1Result
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(
|
||||
stepIdentifier,
|
||||
"PreScoringHydrationPhase1Result")).results.map(_.features)
|
||||
|
||||
val preScoringHydrationPhase2FeatureMaps: Seq[FeatureMap] =
|
||||
previousResult.preScoringHydrationPhase2Result
|
||||
.getOrElse(
|
||||
throw InvalidStepStateException(
|
||||
stepIdentifier,
|
||||
"PreScoringHydrationPhase2Result")).results.map(_.features)
|
||||
/*
|
||||
* If either pre-scoring hydration phase feature map is empty, no need to merge them,
|
||||
* we can just take all non-empty ones.
|
||||
*/
|
||||
if (preScoringHydrationPhase1FeatureMaps.isEmpty) {
|
||||
preScoringHydrationPhase2FeatureMaps
|
||||
} else if (preScoringHydrationPhase2FeatureMaps.isEmpty) {
|
||||
preScoringHydrationPhase1FeatureMaps
|
||||
} else {
|
||||
// No need to check the size in both, since the inputs to both hydration phases are the
|
||||
// same and each phase ensures the number of candidates and ordering matches the input.
|
||||
preScoringHydrationPhase1FeatureMaps.zip(preScoringHydrationPhase2FeatureMaps).map {
|
||||
case (preScoringHydrationPhase1FeatureMap, preScoringHydrationPhasesFeatureMap) =>
|
||||
preScoringHydrationPhase1FeatureMap ++ preScoringHydrationPhasesFeatureMap
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
val ScorersStep =
|
||||
new Step[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Candidate],
|
||||
CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
] {
|
||||
override def identifier: PipelineStepIdentifier = ScoringPipelineConfig.scorersStep
|
||||
|
||||
override def inputAdaptor(
|
||||
query: ScoringPipeline.Inputs[Query],
|
||||
previousResult: ScoringPipelineResult[Candidate]
|
||||
): CandidateFeatureHydratorExecutor.Inputs[Query, Candidate] = {
|
||||
|
||||
val mergedPreScoringFeatureHydrationFeatures: Seq[FeatureMap] =
|
||||
getMergedPreScoringFeatureMap(ScoringPipelineConfig.scorersStep, previousResult)
|
||||
|
||||
val itemCandidates = previousResult.selectorResults
|
||||
.getOrElse(throw InvalidStepStateException(identifier, "SelectionResults"))
|
||||
.selectedCandidates.collect {
|
||||
case itemCandidate: ItemCandidateWithDetails => itemCandidate
|
||||
}
|
||||
|
||||
// If there was no pre-scoring features hydration, no need to re-merge feature maps
|
||||
// and construct a new item candidate
|
||||
val updatedCandidates = if (mergedPreScoringFeatureHydrationFeatures.isEmpty) {
|
||||
itemCandidates
|
||||
} else {
|
||||
itemCandidates.zip(mergedPreScoringFeatureHydrationFeatures).map {
|
||||
case (itemCandidate, preScoringFeatureMap) =>
|
||||
itemCandidate.copy(features = itemCandidate.features ++ preScoringFeatureMap)
|
||||
}
|
||||
}
|
||||
CandidateFeatureHydratorExecutor.Inputs(
|
||||
query.query,
|
||||
updatedCandidates.asInstanceOf[Seq[CandidateWithFeatures[Candidate]]])
|
||||
}
|
||||
|
||||
override lazy val executorArrow: Arrow[
|
||||
CandidateFeatureHydratorExecutor.Inputs[Query, Candidate],
|
||||
CandidateFeatureHydratorExecutorResult[
|
||||
Candidate
|
||||
]
|
||||
] = candidateFeatureHydratorExecutor.arrow(config.scorers.toSeq, context)
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ScoringPipelineResult[Candidate],
|
||||
executorResult: CandidateFeatureHydratorExecutorResult[Candidate]
|
||||
): ScoringPipelineResult[Candidate] =
|
||||
previousPipelineResult.copy(scorerResults = Some(executorResult))
|
||||
}
|
||||
|
||||
val ResultStep =
|
||||
new Step[Seq[CandidateWithFeatures[UniversalNoun[Any]]], Seq[
|
||||
CandidateWithFeatures[UniversalNoun[Any]]
|
||||
]] {
|
||||
override def identifier: PipelineStepIdentifier = ScoringPipelineConfig.resultStep
|
||||
|
||||
override def executorArrow: Arrow[Seq[CandidateWithFeatures[UniversalNoun[Any]]], Seq[
|
||||
CandidateWithFeatures[UniversalNoun[Any]]
|
||||
]] = Arrow.identity
|
||||
|
||||
override def inputAdaptor(
|
||||
query: Inputs[Query],
|
||||
previousResult: ScoringPipelineResult[Candidate]
|
||||
): Seq[CandidateWithFeatures[UniversalNoun[Any]]] = previousResult.selectorResults
|
||||
.getOrElse(throw InvalidStepStateException(identifier, "SelectionResults"))
|
||||
.selectedCandidates.collect {
|
||||
case itemCandidate: ItemCandidateWithDetails => itemCandidate
|
||||
}
|
||||
|
||||
override def resultUpdater(
|
||||
previousPipelineResult: ScoringPipelineResult[Candidate],
|
||||
executorResult: Seq[CandidateWithFeatures[UniversalNoun[Any]]]
|
||||
): ScoringPipelineResult[Candidate] = {
|
||||
val scorerResults: Seq[FeatureMap] = previousPipelineResult.scorerResults
|
||||
.getOrElse(throw InvalidStepStateException(identifier, "ScorerResult")).results.map(
|
||||
_.features)
|
||||
|
||||
val mergedPreScoringFeatureHydrationFeatureMaps: Seq[FeatureMap] =
|
||||
getMergedPreScoringFeatureMap(ScoringPipelineConfig.resultStep, previousPipelineResult)
|
||||
|
||||
val itemCandidates = executorResult.asInstanceOf[Seq[ItemCandidateWithDetails]]
|
||||
val finalFeatureMap = if (mergedPreScoringFeatureHydrationFeatureMaps.isEmpty) {
|
||||
scorerResults
|
||||
} else {
|
||||
scorerResults
|
||||
.zip(mergedPreScoringFeatureHydrationFeatureMaps).map {
|
||||
case (preScoringFeatureMap, scoringFeatureMap) =>
|
||||
preScoringFeatureMap ++ scoringFeatureMap
|
||||
}
|
||||
}
|
||||
|
||||
val finalResults = itemCandidates.zip(finalFeatureMap).map {
|
||||
case (itemCandidate, featureMap) =>
|
||||
ScoredCandidateResult(itemCandidate.candidate.asInstanceOf[Candidate], featureMap)
|
||||
}
|
||||
previousPipelineResult.withResult(finalResults)
|
||||
}
|
||||
}
|
||||
|
||||
val builtSteps = Seq(
|
||||
GatesStep,
|
||||
SelectorsStep,
|
||||
PreScoringFeatureHydrationPhase1Step,
|
||||
PreScoringFeatureHydrationPhase2Step,
|
||||
ScorersStep,
|
||||
ResultStep
|
||||
)
|
||||
|
||||
/** The main execution logic for this Candidate Pipeline. */
|
||||
val finalArrow: Arrow[ScoringPipeline.Inputs[Query], ScoringPipelineResult[Candidate]] =
|
||||
buildCombinedArrowFromSteps(
|
||||
steps = builtSteps,
|
||||
context = context,
|
||||
initialEmptyResult = ScoringPipelineResult.empty,
|
||||
stepsInOrderFromConfig = ScoringPipelineConfig.stepsInOrder
|
||||
)
|
||||
|
||||
val configFromBuilder = config
|
||||
new ScoringPipeline[Query, Candidate] {
|
||||
override private[core] val config: ScoringPipelineConfig[Query, Candidate] = configFromBuilder
|
||||
override val arrow: Arrow[ScoringPipeline.Inputs[Query], ScoringPipelineResult[Candidate]] =
|
||||
finalArrow
|
||||
override val identifier: ScoringPipelineIdentifier = pipelineIdentifier
|
||||
override val alerts: Seq[Alert] = config.alerts
|
||||
override val children: Seq[Component] =
|
||||
allGates ++ config.preScoringFeatureHydrationPhase1 ++ config.preScoringFeatureHydrationPhase2 ++ config.scorers
|
||||
}
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,30 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.scoring
|
||||
|
||||
import com.twitter.finagle.stats.StatsReceiver
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutor
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutor
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutor
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
|
||||
@Singleton
|
||||
class ScoringPipelineBuilderFactory @Inject() (
|
||||
gateExecutor: GateExecutor,
|
||||
selectorExecutor: SelectorExecutor,
|
||||
candidateFeatureHydratorExecutor: CandidateFeatureHydratorExecutor,
|
||||
statsReceiver: StatsReceiver) {
|
||||
|
||||
def get[
|
||||
Query <: PipelineQuery,
|
||||
Candidate <: UniversalNoun[Any]
|
||||
]: ScoringPipelineBuilder[Query, Candidate] = {
|
||||
new ScoringPipelineBuilder[Query, Candidate](
|
||||
gateExecutor,
|
||||
selectorExecutor,
|
||||
candidateFeatureHydratorExecutor,
|
||||
statsReceiver
|
||||
)
|
||||
}
|
||||
}
|
Binary file not shown.
@ -1,131 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.scoring
|
||||
|
||||
import com.twitter.product_mixer.component_library.selector.InsertAppendResults
|
||||
import com.twitter.product_mixer.core.functional_component.common.AllPipelines
|
||||
import com.twitter.product_mixer.core.functional_component.common.alert.Alert
|
||||
import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator
|
||||
import com.twitter.product_mixer.core.functional_component.gate.BaseGate
|
||||
import com.twitter.product_mixer.core.functional_component.scorer.Scorer
|
||||
import com.twitter.product_mixer.core.functional_component.selector.Selector
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifierStack
|
||||
import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfig
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineConfigCompanion
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.timelines.configapi.FSParam
|
||||
import com.twitter.timelines.configapi.decider.DeciderParam
|
||||
|
||||
/**
|
||||
* This is the configuration necessary to generate a Scoring Pipeline. Product code should create a
|
||||
* ScoringPipelineConfig, and then use a ScoringPipelineBuilder to get the final ScoringPipeline which can
|
||||
* process requests.
|
||||
*
|
||||
* @tparam Query - The domain model for the query or request
|
||||
* @tparam Candidate the domain model for the candidate being scored
|
||||
*/
|
||||
trait ScoringPipelineConfig[-Query <: PipelineQuery, Candidate <: UniversalNoun[Any]]
|
||||
extends PipelineConfig {
|
||||
|
||||
override val identifier: ScoringPipelineIdentifier
|
||||
|
||||
/**
|
||||
* When these Params are defined, they will automatically be added as Gates in the pipeline
|
||||
* by the CandidatePipelineBuilder
|
||||
*
|
||||
* The enabled decider param can to be used to quickly disable a Candidate Pipeline via Decider
|
||||
*/
|
||||
val enabledDeciderParam: Option[DeciderParam[Boolean]] = None
|
||||
|
||||
/**
|
||||
* This supported client feature switch param can be used with a Feature Switch to control the
|
||||
* rollout of a new Candidate Pipeline from dogfood to experiment to production
|
||||
*/
|
||||
val supportedClientParam: Option[FSParam[Boolean]] = None
|
||||
|
||||
/** [[BaseGate]]s that are applied sequentially, the pipeline will only run if all the Gates are open */
|
||||
def gates: Seq[BaseGate[Query]] = Seq.empty
|
||||
|
||||
/**
|
||||
* Logic for selecting which candidates to score. Note, this doesn't drop the candidates from
|
||||
* the final result, just whether to score it in this pipeline or not.
|
||||
*/
|
||||
def selectors: Seq[Selector[Query]]
|
||||
|
||||
/**
|
||||
* After selectors are run, you can fetch features for each candidate.
|
||||
* The existing features from previous hydrations are passed in as inputs. You are not expected to
|
||||
* put them into the resulting feature map yourself - they will be merged for you by the platform.
|
||||
*/
|
||||
def preScoringFeatureHydrationPhase1: Seq[BaseCandidateFeatureHydrator[Query, Candidate, _]] =
|
||||
Seq.empty
|
||||
|
||||
/**
|
||||
* A second phase of feature hydration that can be run after selection and after the first phase
|
||||
* of pre-scoring feature hydration. You are not expected to put them into the resulting
|
||||
* feature map yourself - they will be merged for you by the platform.
|
||||
*/
|
||||
def preScoringFeatureHydrationPhase2: Seq[BaseCandidateFeatureHydrator[Query, Candidate, _]] =
|
||||
Seq.empty
|
||||
|
||||
/**
|
||||
* Ranker Function for candidates. Scorers are executed in parallel.
|
||||
* Note: Order does not matter, this could be a Set if Set was covariant over it's type.
|
||||
*/
|
||||
def scorers: Seq[Scorer[Query, Candidate]]
|
||||
|
||||
/**
|
||||
* A pipeline can define a partial function to rescue failures here. They will be treated as failures
|
||||
* from a monitoring standpoint, and cancellation exceptions will always be propagated (they cannot be caught here).
|
||||
*/
|
||||
def failureClassifier: PartialFunction[Throwable, PipelineFailure] = PartialFunction.empty
|
||||
|
||||
/**
|
||||
* Alerts can be used to indicate the pipeline's service level objectives. Alerts and
|
||||
* dashboards will be automatically created based on this information.
|
||||
*/
|
||||
val alerts: Seq[Alert] = Seq.empty
|
||||
|
||||
/**
|
||||
* This method is used by the product mixer framework to build the pipeline.
|
||||
*/
|
||||
private[core] final def build(
|
||||
parentComponentIdentifierStack: ComponentIdentifierStack,
|
||||
builder: ScoringPipelineBuilderFactory
|
||||
): ScoringPipeline[Query, Candidate] =
|
||||
builder.get.build(parentComponentIdentifierStack, this)
|
||||
}
|
||||
|
||||
object ScoringPipelineConfig extends PipelineConfigCompanion {
|
||||
def apply[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]](
|
||||
scorer: Scorer[Query, Candidate]
|
||||
): ScoringPipelineConfig[Query, Candidate] = new ScoringPipelineConfig[Query, Candidate] {
|
||||
override val identifier: ScoringPipelineIdentifier = ScoringPipelineIdentifier(
|
||||
s"ScoreAll${scorer.identifier.name}")
|
||||
|
||||
override val selectors: Seq[Selector[Query]] = Seq(InsertAppendResults(AllPipelines))
|
||||
|
||||
override val scorers: Seq[Scorer[Query, Candidate]] = Seq(scorer)
|
||||
}
|
||||
|
||||
val gatesStep: PipelineStepIdentifier = PipelineStepIdentifier("Gates")
|
||||
val selectorsStep: PipelineStepIdentifier = PipelineStepIdentifier("Selectors")
|
||||
val preScoringFeatureHydrationPhase1Step: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("PreScoringFeatureHydrationPhase1")
|
||||
val preScoringFeatureHydrationPhase2Step: PipelineStepIdentifier =
|
||||
PipelineStepIdentifier("PreScoringFeatureHydrationPhase2")
|
||||
val scorersStep: PipelineStepIdentifier = PipelineStepIdentifier("Scorers")
|
||||
val resultStep: PipelineStepIdentifier = PipelineStepIdentifier("Result")
|
||||
|
||||
/** All the Steps which are executed by a [[ScoringPipeline]] in the order in which they are run */
|
||||
override val stepsInOrder: Seq[PipelineStepIdentifier] = Seq(
|
||||
gatesStep,
|
||||
selectorsStep,
|
||||
preScoringFeatureHydrationPhase1Step,
|
||||
preScoringFeatureHydrationPhase2Step,
|
||||
scorersStep,
|
||||
resultStep
|
||||
)
|
||||
}
|
Binary file not shown.
@ -1,51 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.scoring
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.scorer.ScoredCandidateResult
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineResult
|
||||
import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure
|
||||
import com.twitter.product_mixer.core.service.candidate_feature_hydrator_executor.CandidateFeatureHydratorExecutorResult
|
||||
import com.twitter.product_mixer.core.service.gate_executor.GateExecutorResult
|
||||
import com.twitter.product_mixer.core.service.selector_executor.SelectorExecutorResult
|
||||
|
||||
/**
|
||||
* The Results of every step during the ScoringPipeline process. The end result contains
|
||||
* only the candidates that were actually scored (e.g, not dropped by a filter) with an updated,
|
||||
* combined feature map of all features that were passed in with the candidate plus all features
|
||||
* returned as part of scoring.
|
||||
*/
|
||||
case class ScoringPipelineResult[Candidate <: UniversalNoun[Any]](
|
||||
gateResults: Option[GateExecutorResult],
|
||||
selectorResults: Option[SelectorExecutorResult],
|
||||
preScoringHydrationPhase1Result: Option[CandidateFeatureHydratorExecutorResult[Candidate]],
|
||||
preScoringHydrationPhase2Result: Option[CandidateFeatureHydratorExecutorResult[Candidate]],
|
||||
scorerResults: Option[CandidateFeatureHydratorExecutorResult[
|
||||
Candidate
|
||||
]],
|
||||
failure: Option[PipelineFailure],
|
||||
result: Option[Seq[ScoredCandidateResult[Candidate]]])
|
||||
extends PipelineResult[Seq[ScoredCandidateResult[Candidate]]] {
|
||||
override val resultSize: Int = result.map(_.size).getOrElse(0)
|
||||
|
||||
override def withFailure(
|
||||
failure: PipelineFailure
|
||||
): ScoringPipelineResult[Candidate] =
|
||||
copy(failure = Some(failure))
|
||||
override def withResult(
|
||||
result: Seq[ScoredCandidateResult[Candidate]]
|
||||
): ScoringPipelineResult[Candidate] =
|
||||
copy(result = Some(result))
|
||||
}
|
||||
|
||||
object ScoringPipelineResult {
|
||||
def empty[Candidate <: UniversalNoun[Any]]: ScoringPipelineResult[Candidate] =
|
||||
ScoringPipelineResult(
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None,
|
||||
None
|
||||
)
|
||||
}
|
@ -1,33 +0,0 @@
|
||||
scala_library(
|
||||
sources = [
|
||||
"*.scala",
|
||||
],
|
||||
compiler_option_sets = ["fatal_warnings"],
|
||||
platform = "java8",
|
||||
strict_deps = True,
|
||||
tags = ["bazel-compatible"],
|
||||
dependencies = [
|
||||
"product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/asyncfeaturemap",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator:decoration",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline:query",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service",
|
||||
],
|
||||
exports = [
|
||||
"product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/asyncfeaturemap",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator:decoration",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline:query",
|
||||
"product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service",
|
||||
],
|
||||
)
|
Binary file not shown.
Binary file not shown.
@ -1,9 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.state
|
||||
|
||||
import com.twitter.product_mixer.core.feature.featuremap.asyncfeaturemap.AsyncFeatureMap
|
||||
|
||||
trait HasAsyncFeatureMap[State] {
|
||||
def asyncFeatureMap: AsyncFeatureMap
|
||||
|
||||
private[core] def addAsyncFeatureMap(newFeatureMap: AsyncFeatureMap): State
|
||||
}
|
Binary file not shown.
@ -1,8 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.state
|
||||
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
|
||||
trait HasCandidates[Candidate <: UniversalNoun[Any], T] {
|
||||
def candidates: Seq[Candidate]
|
||||
def updateCandidates(newCandidates: Seq[Candidate]): T
|
||||
}
|
Binary file not shown.
@ -1,11 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.state
|
||||
|
||||
import com.twitter.product_mixer.core.functional_component.decorator.Decoration
|
||||
import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails
|
||||
|
||||
trait HasCandidatesWithDetails[T] {
|
||||
def candidatesWithDetails: Seq[CandidateWithDetails]
|
||||
def updateCandidatesWithDetails(newCandidates: Seq[CandidateWithDetails]): T
|
||||
|
||||
def updateDecorations(decoration: Seq[Decoration]): T
|
||||
}
|
Binary file not shown.
@ -1,9 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.state
|
||||
|
||||
import com.twitter.product_mixer.core.model.common.CandidateWithFeatures
|
||||
import com.twitter.product_mixer.core.model.common.UniversalNoun
|
||||
|
||||
trait HasCandidatesWithFeatures[Candidate <: UniversalNoun[Any], T] {
|
||||
def candidatesWithFeatures: Seq[CandidateWithFeatures[Candidate]]
|
||||
def updateCandidatesWithFeatures(newCandidates: Seq[CandidateWithFeatures[Candidate]]): T
|
||||
}
|
Binary file not shown.
@ -1,13 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.state
|
||||
|
||||
import com.twitter.product_mixer.core.model.common.identifier.PipelineStepIdentifier
|
||||
import com.twitter.product_mixer.core.service.ExecutorResult
|
||||
import scala.collection.immutable.ListMap
|
||||
|
||||
trait HasExecutorResults[State] {
|
||||
// We use a list map to maintain the insertion order
|
||||
val executorResultsByPipelineStep: ListMap[PipelineStepIdentifier, ExecutorResult]
|
||||
private[pipeline] def setExecutorResults(
|
||||
newMap: ListMap[PipelineStepIdentifier, ExecutorResult]
|
||||
): State
|
||||
}
|
Binary file not shown.
@ -1,7 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.state
|
||||
|
||||
import com.twitter.timelines.configapi.Params
|
||||
|
||||
trait HasParams {
|
||||
def params: Params
|
||||
}
|
Binary file not shown.
@ -1,8 +0,0 @@
|
||||
package com.twitter.product_mixer.core.pipeline.state
|
||||
|
||||
import com.twitter.product_mixer.core.pipeline.PipelineQuery
|
||||
|
||||
trait HasQuery[Query <: PipelineQuery, T] {
|
||||
def query: Query
|
||||
def updateQuery(query: Query): T
|
||||
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user