Skip to content

Commit

Permalink
Add WorkflowTracer Through some Runtime Internals
Browse files Browse the repository at this point in the history
  • Loading branch information
steve-the-edwards committed Dec 5, 2024
1 parent cd7485e commit cdd7c76
Show file tree
Hide file tree
Showing 23 changed files with 259 additions and 62 deletions.
13 changes: 13 additions & 0 deletions workflow-core/api/workflow-core.api
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ public abstract interface class com/squareup/workflow1/BaseRenderContext {
public abstract fun eventHandler (Ljava/lang/String;Lkotlin/jvm/functions/Function8;)Lkotlin/jvm/functions/Function7;
public abstract fun eventHandler (Ljava/lang/String;Lkotlin/jvm/functions/Function9;)Lkotlin/jvm/functions/Function8;
public abstract fun getActionSink ()Lcom/squareup/workflow1/Sink;
public abstract fun getWorkflowTracer ()Lcom/squareup/workflow1/WorkflowTracer;
public abstract fun renderChild (Lcom/squareup/workflow1/Workflow;Ljava/lang/Object;Ljava/lang/String;Lkotlin/jvm/functions/Function1;)Ljava/lang/Object;
public abstract fun runningSideEffect (Ljava/lang/String;Lkotlin/jvm/functions/Function2;)V
}
Expand Down Expand Up @@ -167,6 +168,7 @@ public final class com/squareup/workflow1/StatefulWorkflow$RenderContext : com/s
public fun eventHandler (Ljava/lang/String;Lkotlin/jvm/functions/Function8;)Lkotlin/jvm/functions/Function7;
public fun eventHandler (Ljava/lang/String;Lkotlin/jvm/functions/Function9;)Lkotlin/jvm/functions/Function8;
public fun getActionSink ()Lcom/squareup/workflow1/Sink;
public fun getWorkflowTracer ()Lcom/squareup/workflow1/WorkflowTracer;
public fun renderChild (Lcom/squareup/workflow1/Workflow;Ljava/lang/Object;Ljava/lang/String;Lkotlin/jvm/functions/Function1;)Ljava/lang/Object;
public fun runningSideEffect (Ljava/lang/String;Lkotlin/jvm/functions/Function2;)V
}
Expand All @@ -192,6 +194,7 @@ public final class com/squareup/workflow1/StatelessWorkflow$RenderContext : com/
public fun eventHandler (Ljava/lang/String;Lkotlin/jvm/functions/Function8;)Lkotlin/jvm/functions/Function7;
public fun eventHandler (Ljava/lang/String;Lkotlin/jvm/functions/Function9;)Lkotlin/jvm/functions/Function8;
public fun getActionSink ()Lcom/squareup/workflow1/Sink;
public fun getWorkflowTracer ()Lcom/squareup/workflow1/WorkflowTracer;
public fun renderChild (Lcom/squareup/workflow1/Workflow;Ljava/lang/Object;Ljava/lang/String;Lkotlin/jvm/functions/Function1;)Ljava/lang/Object;
public fun runningSideEffect (Ljava/lang/String;Lkotlin/jvm/functions/Function2;)V
}
Expand Down Expand Up @@ -306,6 +309,16 @@ public final class com/squareup/workflow1/WorkflowOutput {
public fun toString ()Ljava/lang/String;
}

public abstract interface class com/squareup/workflow1/WorkflowTracer {
public abstract fun beginSection (Ljava/lang/String;)V
public abstract fun endSection ()V
}

public final class com/squareup/workflow1/WorkflowTracerKt {
public static final fun trace (Lcom/squareup/workflow1/WorkflowTracer;Ljava/lang/String;Lkotlin/jvm/functions/Function0;)Ljava/lang/Object;
public static final fun trace (Lcom/squareup/workflow1/WorkflowTracer;Lkotlin/jvm/functions/Function0;Lkotlin/jvm/functions/Function0;)Ljava/lang/Object;
}

public final class com/squareup/workflow1/Workflows {
public static final fun RenderContext (Lcom/squareup/workflow1/BaseRenderContext;Lcom/squareup/workflow1/StatefulWorkflow;)Lcom/squareup/workflow1/StatefulWorkflow$RenderContext;
public static final fun RenderContext (Lcom/squareup/workflow1/BaseRenderContext;Lcom/squareup/workflow1/StatelessWorkflow;)Lcom/squareup/workflow1/StatelessWorkflow$RenderContext;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,8 @@ public interface BaseRenderContext<out PropsT, StateT, in OutputT> {
*/
public val actionSink: Sink<WorkflowAction<PropsT, StateT, OutputT>>

public val workflowTracer: WorkflowTracer?

/**
* Ensures [child] is running as a child of this workflow, and returns the result of its
* `render` method.
Expand Down Expand Up @@ -437,6 +439,8 @@ internal fun <T, PropsT, StateT, OutputT>
key: String = "",
handler: (T) -> WorkflowAction<PropsT, StateT, OutputT>
) {
val workerWorkflow = WorkerWorkflow<T>(workerType, key)
val workerWorkflow = workflowTracer.trace({ "Create WorkerWorkflow($workerType,$key)" }) {
WorkerWorkflow<T>(workerType, key)
}
renderChild(workerWorkflow, props = worker, key = key, handler = handler)
}
Original file line number Diff line number Diff line change
Expand Up @@ -30,11 +30,16 @@ import kotlin.reflect.KType
*/
internal class WorkerWorkflow<OutputT>(
val workerType: KType,
private val key: String
private val key: String,
workflowTracer: WorkflowTracer? = null
) : StatefulWorkflow<Worker<OutputT>, Int, OutputT, Unit>(),
ImpostorWorkflow {

override val realIdentifier: WorkflowIdentifier = unsnapshottableIdentifier(workerType)
override val realIdentifier: WorkflowIdentifier =
workflowTracer.trace({ "workerIdentifier($workerType)" }) {
unsnapshottableIdentifier(workerType)
}

override fun describeRealIdentifier(): String = workerType.toString()

override fun initialState(
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
package com.squareup.workflow1

/**
* This is a very simple tracing interface that can be passed into a workflow runtime in order
* to inject span tracing throughout the workflow core and runtime internals.
*/
public interface WorkflowTracer {
public fun beginSection(label: String): Unit
public fun endSection(): Unit
}

/**
* Convenience function to wrap [block] with a trace span as defined by [WorkflowTracer].
* Only calls [label] if there is an active [WorkflowTracer] use this for any label other than
* a constant.
*/
public inline fun <T> WorkflowTracer?.trace(label: () -> String, block: () -> T): T {
val optimizedLabel = if (this !== null) {
label()
} else {
""
}
return trace(optimizedLabel, block)
}

/**
* Convenience function to wrap [block] with a trace span as defined by [WorkflowTracer].
*/
public inline fun <T> WorkflowTracer?.trace(label: String, block: () -> T): T {
this?.beginSection(label)
try {
return block()
} finally {
this?.endSection()
}
}
9 changes: 7 additions & 2 deletions workflow-runtime/api/workflow-runtime.api
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,8 @@ public final class com/squareup/workflow1/NoopWorkflowInterceptor : com/squareup
}

public final class com/squareup/workflow1/RenderWorkflowKt {
public static final fun renderWorkflowIn (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Lkotlinx/coroutines/flow/StateFlow;Lcom/squareup/workflow1/TreeSnapshot;Ljava/util/List;Ljava/util/Set;Lkotlin/jvm/functions/Function2;)Lkotlinx/coroutines/flow/StateFlow;
public static synthetic fun renderWorkflowIn$default (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Lkotlinx/coroutines/flow/StateFlow;Lcom/squareup/workflow1/TreeSnapshot;Ljava/util/List;Ljava/util/Set;Lkotlin/jvm/functions/Function2;ILjava/lang/Object;)Lkotlinx/coroutines/flow/StateFlow;
public static final fun renderWorkflowIn (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Lkotlinx/coroutines/flow/StateFlow;Lcom/squareup/workflow1/TreeSnapshot;Ljava/util/List;Ljava/util/Set;Lcom/squareup/workflow1/WorkflowTracer;Lkotlin/jvm/functions/Function2;)Lkotlinx/coroutines/flow/StateFlow;
public static synthetic fun renderWorkflowIn$default (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Lkotlinx/coroutines/flow/StateFlow;Lcom/squareup/workflow1/TreeSnapshot;Ljava/util/List;Ljava/util/Set;Lcom/squareup/workflow1/WorkflowTracer;Lkotlin/jvm/functions/Function2;ILjava/lang/Object;)Lkotlinx/coroutines/flow/StateFlow;
}

public final class com/squareup/workflow1/RenderingAndSnapshot {
Expand Down Expand Up @@ -104,5 +104,10 @@ public abstract interface class com/squareup/workflow1/WorkflowInterceptor$Workf
public abstract fun getRenderKey ()Ljava/lang/String;
public abstract fun getRuntimeConfig ()Ljava/util/Set;
public abstract fun getSessionId ()J
public abstract fun getWorkflowTracer ()Lcom/squareup/workflow1/WorkflowTracer;
}

public final class com/squareup/workflow1/internal/WorkflowLoggingKt {
public static final fun toLoggingName (Lcom/squareup/workflow1/WorkflowIdentifier;)Ljava/lang/String;
}

Original file line number Diff line number Diff line change
Expand Up @@ -110,12 +110,21 @@ public fun <PropsT, OutputT, RenderingT> renderWorkflowIn(
initialSnapshot: TreeSnapshot? = null,
interceptors: List<WorkflowInterceptor> = emptyList(),
runtimeConfig: RuntimeConfig = RuntimeConfigOptions.DEFAULT_CONFIG,
workflowTracer: WorkflowTracer? = null,
onOutput: suspend (OutputT) -> Unit
): StateFlow<RenderingAndSnapshot<RenderingT>> {
val chainedInterceptor = interceptors.chained()

val runner =
WorkflowRunner(scope, workflow, props, initialSnapshot, chainedInterceptor, runtimeConfig)
WorkflowRunner(
scope,
workflow,
props,
initialSnapshot,
chainedInterceptor,
runtimeConfig,
workflowTracer
)

// Rendering is synchronous, so we can run the first render pass before launching the runtime
// coroutine to calculate the initial rendering.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,9 @@ public interface WorkflowInterceptor {

/** The [RuntimeConfig] of the runtime this session is executing in. */
public val runtimeConfig: RuntimeConfig

/** The optional [WorkflowTracer] of the runtime this session is executing in. */
public val workflowTracer: WorkflowTracer?
}

/**
Expand Down Expand Up @@ -308,6 +311,7 @@ private class InterceptedRenderContext<P, S, O>(
private val interceptor: RenderContextInterceptor<P, S, O>
) : BaseRenderContext<P, S, O>, Sink<WorkflowAction<P, S, O>> {
override val actionSink: Sink<WorkflowAction<P, S, O>> get() = this
override val workflowTracer: WorkflowTracer? = baseRenderContext.workflowTracer

override fun send(value: WorkflowAction<P, S, O>) {
interceptor.onActionSent(value) { interceptedAction ->
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,13 +4,15 @@ import com.squareup.workflow1.BaseRenderContext
import com.squareup.workflow1.Sink
import com.squareup.workflow1.Workflow
import com.squareup.workflow1.WorkflowAction
import com.squareup.workflow1.WorkflowTracer
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.channels.SendChannel

internal class RealRenderContext<out PropsT, StateT, OutputT>(
private val renderer: Renderer<PropsT, StateT, OutputT>,
private val sideEffectRunner: SideEffectRunner,
private val eventActionsChannel: SendChannel<WorkflowAction<PropsT, StateT, OutputT>>
private val eventActionsChannel: SendChannel<WorkflowAction<PropsT, StateT, OutputT>>,
override val workflowTracer: WorkflowTracer?
) : BaseRenderContext<PropsT, StateT, OutputT>, Sink<WorkflowAction<PropsT, StateT, OutputT>> {

interface Renderer<PropsT, StateT, OutputT> {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,9 @@ import com.squareup.workflow1.Workflow
import com.squareup.workflow1.WorkflowAction
import com.squareup.workflow1.WorkflowInterceptor
import com.squareup.workflow1.WorkflowInterceptor.WorkflowSession
import com.squareup.workflow1.WorkflowTracer
import com.squareup.workflow1.identifier
import com.squareup.workflow1.trace
import kotlinx.coroutines.selects.SelectBuilder
import kotlin.coroutines.CoroutineContext

Expand Down Expand Up @@ -91,6 +93,7 @@ internal class SubtreeManager<PropsT, StateT, OutputT>(
childResult: ActionApplied<*>
) -> ActionProcessingResult,
private val runtimeConfig: RuntimeConfig,
private val workflowTracer: WorkflowTracer?,
private val workflowSession: WorkflowSession? = null,
private val interceptor: WorkflowInterceptor = NoopWorkflowInterceptor,
private val idCounter: IdCounter? = null
Expand Down Expand Up @@ -121,17 +124,22 @@ internal class SubtreeManager<PropsT, StateT, OutputT>(
handler: (ChildOutputT) -> WorkflowAction<PropsT, StateT, OutputT>
): ChildRenderingT {
// Prevent duplicate workflows with the same key.
children.forEachStaging {
require(!(it.matches(child, key))) {
"Expected keys to be unique for ${child.identifier}: key=\"$key\""
workflowTracer.trace({ "CheckingUniqueFor:${child.identifier.toLoggingName()}" }) {
children.forEachStaging {
require(!(it.matches(child, key))) {
"Expected keys to be unique for ${child.identifier}: key=\"$key\""
}
}
}

// Start tracking this case so we can be ready to render it.
val stagedChild = children.retainOrCreate(
predicate = { it.matches(child, key) },
create = { createChildNode(child, props, key, handler) }
)
val stagedChild =
workflowTracer.trace({ "RetainingChildrenFor:${child.identifier.toLoggingName()}" }) {
children.retainOrCreate(
predicate = { it.matches(child, key) },
create = { createChildNode(child, props, key, handler) }
)
}
stagedChild.setHandler(handler)
return stagedChild.render(child.asStatefulWorkflow(), props)
}
Expand Down Expand Up @@ -188,6 +196,7 @@ internal class SubtreeManager<PropsT, StateT, OutputT>(
snapshot = childTreeSnapshots,
baseContext = contextForChildren,
runtimeConfig = runtimeConfig,
workflowTracer = workflowTracer,
emitAppliedActionToParent = ::acceptChildActionResult,
parent = workflowSession,
interceptor = interceptor,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
package com.squareup.workflow1.internal

import com.squareup.workflow1.WorkflowIdentifier
import com.squareup.workflow1.WorkflowIdentifierType.Snapshottable
import com.squareup.workflow1.WorkflowIdentifierType.Unsnapshottable
import kotlin.reflect.KClass
import kotlin.reflect.KType

public fun WorkflowIdentifier.toLoggingName(): String {
return when (val type = getRealIdentifierType()) {
is Snapshottable -> type.kClass?.toLoggingName() ?: type.typeName
is Unsnapshottable -> type.kType.toLoggingName()
}
}

private fun KType.toLoggingName(): String {
if (classifier == null) return toString()

val classifierName = when (val c = classifier) {
is KClass<*> -> c.toLoggingName()
else -> toString()
}

val params = arguments.map { projection ->
when (val type = projection.type) {
is KType -> type.toLoggingName()
else -> "*"
}
}

return if (params.isEmpty()) classifierName else "$classifierName<${params.joinToString(", ")}>"
}

private fun KClass<*>.toLoggingName(): String {
return simpleName ?: toString()
}
Original file line number Diff line number Diff line change
Expand Up @@ -14,9 +14,11 @@ import com.squareup.workflow1.WorkflowExperimentalApi
import com.squareup.workflow1.WorkflowIdentifier
import com.squareup.workflow1.WorkflowInterceptor
import com.squareup.workflow1.WorkflowInterceptor.WorkflowSession
import com.squareup.workflow1.WorkflowTracer
import com.squareup.workflow1.applyTo
import com.squareup.workflow1.intercept
import com.squareup.workflow1.internal.RealRenderContext.SideEffectRunner
import com.squareup.workflow1.trace
import kotlinx.coroutines.CancellationException
import kotlinx.coroutines.CoroutineName
import kotlinx.coroutines.CoroutineScope
Expand Down Expand Up @@ -51,6 +53,7 @@ internal class WorkflowNode<PropsT, StateT, OutputT, RenderingT>(
baseContext: CoroutineContext,
// Providing default value so we don't need to specify in test.
override val runtimeConfig: RuntimeConfig = RuntimeConfigOptions.DEFAULT_CONFIG,
override val workflowTracer: WorkflowTracer? = null,
private val emitAppliedActionToParent: (ActionApplied<OutputT>) -> ActionProcessingResult =
{ it },
override val parent: WorkflowSession? = null,
Expand All @@ -74,6 +77,7 @@ internal class WorkflowNode<PropsT, StateT, OutputT, RenderingT>(
contextForChildren = coroutineContext,
emitActionToParent = ::applyAction,
runtimeConfig = runtimeConfig,
workflowTracer = workflowTracer,
workflowSession = this,
interceptor = interceptor,
idCounter = idCounter
Expand All @@ -87,7 +91,8 @@ internal class WorkflowNode<PropsT, StateT, OutputT, RenderingT>(
private val baseRenderContext = RealRenderContext(
renderer = subtreeManager,
sideEffectRunner = this,
eventActionsChannel = eventActionsChannel
eventActionsChannel = eventActionsChannel,
workflowTracer = workflowTracer,
)
private val context = RenderContext(baseRenderContext, workflow)

Expand Down Expand Up @@ -212,12 +217,14 @@ internal class WorkflowNode<PropsT, StateT, OutputT, RenderingT>(
.render(props, state, context)
baseRenderContext.freeze()

// Tear down workflows and workers that are obsolete.
subtreeManager.commitRenderedChildren()
// Side effect jobs are launched lazily, since they can send actions to the sink, and can only
// be started after context is frozen.
sideEffects.forEachStaging { it.job.start() }
sideEffects.commitStaging { it.job.cancel() }
workflowTracer.trace({ "UpdateRuntimeTree:$id" }) {
// Tear down workflows and workers that are obsolete.
subtreeManager.commitRenderedChildren()
// Side effect jobs are launched lazily, since they can send actions to the sink, and can only
// be started after context is frozen.
sideEffects.forEachStaging { it.job.start() }
sideEffects.commitStaging { it.job.cancel() }
}

return rendering
}
Expand Down Expand Up @@ -261,8 +268,10 @@ internal class WorkflowNode<PropsT, StateT, OutputT, RenderingT>(
key: String,
sideEffect: suspend CoroutineScope.() -> Unit
): SideEffectNode {
val scope = this + CoroutineName("sideEffect[$key] for $id")
val job = scope.launch(start = LAZY, block = sideEffect)
return SideEffectNode(key, job)
return workflowTracer.trace({ "CreateSideEffectNode:$key" }) {
val scope = this + CoroutineName("sideEffect[$key] for $id")
val job = scope.launch(start = LAZY, block = sideEffect)
SideEffectNode(key, job)
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import com.squareup.workflow1.TreeSnapshot
import com.squareup.workflow1.Workflow
import com.squareup.workflow1.WorkflowExperimentalRuntime
import com.squareup.workflow1.WorkflowInterceptor
import com.squareup.workflow1.WorkflowTracer
import kotlinx.coroutines.CancellationException
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.DelicateCoroutinesApi
Expand All @@ -28,7 +29,8 @@ internal class WorkflowRunner<PropsT, OutputT, RenderingT>(
props: StateFlow<PropsT>,
snapshot: TreeSnapshot?,
private val interceptor: WorkflowInterceptor,
private val runtimeConfig: RuntimeConfig
private val runtimeConfig: RuntimeConfig,
private val workflowTracer: WorkflowTracer?
) {
private val workflow = protoWorkflow.asStatefulWorkflow()
private val idCounter = IdCounter()
Expand All @@ -55,6 +57,7 @@ internal class WorkflowRunner<PropsT, OutputT, RenderingT>(
snapshot = snapshot,
baseContext = scope.coroutineContext,
runtimeConfig = runtimeConfig,
workflowTracer = workflowTracer,
interceptor = interceptor,
idCounter = idCounter
)
Expand Down
Loading

0 comments on commit cdd7c76

Please sign in to comment.