go - 在 Cadence 工作流程中的循环内调用相同的事件

标签 go workflow cadence-workflow temporal-workflow

我在 cadence 工作流程中有一个问题,我们可以在 for 循环内使用不同的输入调用相同的事件吗?该代码是确定性的吗?如果执行工作流的工作线程在执行过程中停止并稍后重新启动,那么 cadence 重新构建工作流时是否能够重播事件。

例如,我有以下代码。

   func init() {
    workflow.RegisterWithOptions(SampleWorkFlow, workflow.RegisterOptions{Name: "SampleWorkFlow"})
    activity.RegisterWithOptions(SampleActivity, activity.RegisterOptions{Name: "SampleActivity"})
    activity.RegisterWithOptions(SecondActivity, activity.RegisterOptions{Name: "SecondActivity"})
}

// SampleWorkFlow comment
func SampleWorkFlow(ctx workflow.Context, input string) error {

    fmt.Println("Workflow started")
    ctx = workflow.WithTaskList(ctx, sampleTaskList)
    ctx = workflow.WithActivityOptions(ctx, conf.ActivityOptions)

    var result string
    err := workflow.ExecuteActivity(ctx, "SampleActivity", input, "string-value").Get(ctx, &result)
    if err != nil {
        return err
    }

    for i := 1; i <= 10; i++ {
        value := i
        workflow.Go(ctx, func(ctx workflow.Context) {
            err := workflow.ExecuteActivity(ctx, "SecondActivity", input, value).Get(ctx, &result)
            if err != nil {
                log.Println("err=", err)
            }
        })
    }

    return nil

}

// SampleActivity comment
func SampleActivity(ctx context.Context, value, v1 string) (string, error) {
    fmt.Println("Sample activity start")
    for i := 0; i <= 10; i++ {
        fmt.Println(i)
    }
    return "Hello " + value, nil
}

// SecondActivity comment
func SecondActivity(ctx context.Context, value int) (string, error) {

    fmt.Println("Second  activity start")

    fmt.Println("value=", value)
    fmt.Println("Second activity going to end")
    return "Hello " + fmt.Sprintf("%d", value), nil
}

这里,第二个事件在 for 循环内并行调用。 我的第一个问题是这段代码是确定性的吗?

假设循环迭代 5 次后,当 i =5 时,执行此工作流程的工作人员终止,如果工作流程在另一个工作流程中启动,那么 cadence 是否能够重播事件 worker ?

你能回答我的问题吗?

最佳答案

是的,这段代码是确定性的。它不会调用任何非确定性操作(例如随机或 UUID 生成)并使用 workflow.Go 启动 goroutine。所以它是确定性的。代码的复杂性并不影响其确定性。

不相关的 nit。 无需在示例中使用 goroutine,因为通过返回 Future,ExecuteActivity 调用已经是非阻塞的。 因此示例可以简化为:

func SampleWorkFlow(ctx workflow.Context, input string) error {

    fmt.Println("Workflow started")
    ctx = workflow.WithTaskList(ctx, sampleTaskList)
    ctx = workflow.WithActivityOptions(ctx, conf.ActivityOptions)

    var result string
    err := workflow.ExecuteActivity(ctx, "SampleActivity", input, "string-value").Get(ctx, &result)
    if err != nil {
        return err
    }

    for i := 1; i <= 10; i++ {
       workflow.ExecuteActivity(ctx, "SecondActivity", input, i)
    }
    return nil
}

请注意,此示例仍然可能不会按照您期望的方式执行,因为它会在不等待事件完成的情况下完成工作流。所以这些事件根本就不会开始。

以下是等待事件完成的代码:

func SampleWorkFlow(ctx workflow.Context, input string) error {

    fmt.Println("Workflow started")
    ctx = workflow.WithTaskList(ctx, sampleTaskList)
    ctx = workflow.WithActivityOptions(ctx, conf.ActivityOptions)

    var result string
    err := workflow.ExecuteActivity(ctx, "SampleActivity", input, "string-value").Get(ctx, &result)
    if err != nil {
        return err
    }
    var results []workflow.Future
    for i := 1; i <= 10; i++ {
        future := workflow.ExecuteActivity(ctx, "SecondActivity", input, i)
        results = append(results, future)
    }
    for i := 0; i < 10; i++ {
        var result string
        err := results[i].Get(ctx, &result)
        if err != nil {
            log.Println("err=", err)
        }
    }
    return nil
}

关于go - 在 Cadence 工作流程中的循环内调用相同的事件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64068770/

相关文章:

http - Go 中的 request.param(...) 在哪里

go - 如何使用 golang 使用 "SCAN"而不是 "KEYS"从 redis 扫描键

testing - 带有测试服务器的 Git 工作流程

go - 如何创建自定义工作流定义?

go - 脚踏车停运时的脚踏车工作流程

go - 有没有一种方法可以将字段添加到将提供给工作流中所有事件的工作流上下文中?

azure - 如何使用 Go SDK 在 Azure AD 中创建新组?

go - 使用 Go 的 HTTP 服务器进行生产

使用集中式工作流的 Git 版本编号

workflow - 有人成功使用CMU Sphinx进行语音识别编程吗