Skip to content

Commit 81312e9

Browse files
mikeeearturotrenardyaron2Eileen-Yu
authored
merge release-1.12 into main (#703)
* Adding the name of the failing activity. For more detail (#678) Signed-off-by: arturo <[email protected]> * add deadlettertopic support to non-streaming subscriptions (#685) * add deadlettertopic support to non-streaming subscriptions Signed-off-by: yaron2 <[email protected]> * fix tests Signed-off-by: yaron2 <[email protected]> --------- Signed-off-by: yaron2 <[email protected]> * Pick #674 and bump to rc13 (#686) * Bump gover, tag, x/deps and dapr (#674) * release: bump to rc2 Signed-off-by: mikeee <[email protected]> * chore: upgrade x/net and x/crypto Signed-off-by: mikeee <[email protected]> * release: bump go to 1.23.5 and dapr to rc5 Signed-off-by: mikeee <[email protected]> * ci: bump validation workflow versions Signed-off-by: mikeee <[email protected]> * bump cli and runtime to latest rc Signed-off-by: Mike Nguyen <[email protected]> * chore: bump dapr to rc7 and dt-go to head Signed-off-by: Mike Nguyen <[email protected]> * chore: bump to rc8 Signed-off-by: mikeee <[email protected]> * chore(release): bump to latest Signed-off-by: mikeee <[email protected]> --------- Signed-off-by: mikeee <[email protected]> Signed-off-by: Mike Nguyen <[email protected]> * chore: bump cli and runtime vers Signed-off-by: Mike Nguyen <[email protected]> * chore: bump to rc13 Signed-off-by: Mike Nguyen <[email protected]> --------- Signed-off-by: mikeee <[email protected]> Signed-off-by: Mike Nguyen <[email protected]> * feat: reconnect stream when grpc code is unknown / unavailable (#692) * feat: reconnect stream when grpc code is unknown / unavailable Signed-off-by: Eileen Yu <[email protected]> * log error for closing stream Signed-off-by: Eileen Yu <[email protected]> --------- Signed-off-by: Eileen Yu <[email protected]> * ci: include pre-release label for RCs (#675) * ci: include pre-release label for RCs Signed-off-by: Mike Nguyen <[email protected]> * ci: enumerate if statements Signed-off-by: Mike Nguyen <[email protected]> --------- Signed-off-by: Mike Nguyen <[email protected]> * docs(sdk): add basic workflow example (#691) * docs(sdk): add basic workflow example Signed-off-by: Mike Nguyen <[email protected]> * docs: fix indentations Signed-off-by: Mike Nguyen <[email protected]> --------- Signed-off-by: Mike Nguyen <[email protected]> * update conversation api field name (#695) Signed-off-by: yaron2 <[email protected]> * fix(examples): update deprecated flag (#689) * fix(examples): update deprecated flag Signed-off-by: Mike Nguyen <[email protected]> * ci: test cli PR Signed-off-by: Mike Nguyen <[email protected]> * test(service): bump body size to 41Mi Signed-off-by: Mike Nguyen <[email protected]> * chore: bump cli to rc6 and runtime to rc16 Signed-off-by: Mike Nguyen <[email protected]> --------- Signed-off-by: Mike Nguyen <[email protected]> Co-authored-by: Yaron Schneider <[email protected]> * release: v1.12.0 version (#700) Signed-off-by: Mike Nguyen <[email protected]> * ci: revert rc tests (#701) Signed-off-by: Mike Nguyen <[email protected]> --------- Signed-off-by: arturo <[email protected]> Signed-off-by: yaron2 <[email protected]> Signed-off-by: mikeee <[email protected]> Signed-off-by: Mike Nguyen <[email protected]> Signed-off-by: Eileen Yu <[email protected]> Co-authored-by: Arturo Trenard <[email protected]> Co-authored-by: Yaron Schneider <[email protected]> Co-authored-by: Eileen Yu <[email protected]>
1 parent c81a381 commit 81312e9

File tree

18 files changed

+259
-83
lines changed

18 files changed

+259
-83
lines changed

.github/workflows/release-on-tag.yaml

+14-1
Original file line numberDiff line numberDiff line change
@@ -38,8 +38,9 @@ jobs:
3838
run: |
3939
echo "RELEASE_VERSION=$(echo ${GITHUB_REF:10})" >> $GITHUB_ENV
4040
41-
- name: Release
41+
- name: Release Main
4242
uses: actions/create-release@v1
43+
if: ${{ !contains(github.ref , 'rc') }}
4344
env:
4445
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
4546
with:
@@ -49,6 +50,18 @@ jobs:
4950
draft: false
5051
prerelease: false
5152

53+
- name: Release RC
54+
uses: actions/create-release@v1
55+
if: ${{ contains(github.ref, 'rc') }}
56+
env:
57+
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
58+
with:
59+
tag_name: ${{ github.ref }}
60+
release_name: Release ${{ github.ref }}
61+
body: Automatic Go Dapr client release
62+
draft: false
63+
prerelease: true
64+
5265
- name: Notify
5366
uses: rjstone/discord-webhook-notify@v1
5467
with:

.github/workflows/validate_examples.yaml

+2-2
Original file line numberDiff line numberDiff line change
@@ -38,9 +38,9 @@ jobs:
3838
CHECKOUT_REF: ${{ github.ref }}
3939
outputs:
4040
DAPR_INSTALL_URL: ${{ env.DAPR_INSTALL_URL }}
41-
DAPR_CLI_VER: 1.15.0-rc.4
41+
DAPR_CLI_VER: ${{ steps.outputs.outputs.DAPR_CLI_VER }}
4242
DAPR_CLI_REF: ${{ steps.outputs.outputs.DAPR_CLI_REF }}
43-
DAPR_RUNTIME_VER: 1.15.0-rc.9
43+
DAPR_RUNTIME_VER: ${{ steps.outputs.outputs.DAPR_RUNTIME_VER }}
4444
CHECKOUT_REPO: ${{ steps.outputs.outputs.CHECKOUT_REPO }}
4545
CHECKOUT_REF: ${{ steps.outputs.outputs.CHECKOUT_REF }}
4646
DAPR_REF: ${{ steps.outputs.outputs.DAPR_REF }}

client/conversation.go

+3-3
Original file line numberDiff line numberDiff line change
@@ -44,8 +44,8 @@ type conversationRequestOption func(request *conversationRequest)
4444

4545
// ConversationInput defines a single input.
4646
type ConversationInput struct {
47-
// The string to send to the llm.
48-
Message string
47+
// The content to send to the llm.
48+
Content string
4949
// The role of the message.
5050
Role *string
5151
// Whether to Scrub PII from the input
@@ -104,7 +104,7 @@ func (c *GRPCClient) ConverseAlpha1(ctx context.Context, req conversationRequest
104104
cinputs := make([]*runtimev1pb.ConversationInput, len(req.inputs))
105105
for i, in := range req.inputs {
106106
cinputs[i] = &runtimev1pb.ConversationInput{
107-
Message: in.Message,
107+
Content: in.Content,
108108
Role: in.Role,
109109
ScrubPII: in.ScrubPII,
110110
}

client/subscribe.go

+102-37
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,9 @@ import (
2323
"sync"
2424
"sync/atomic"
2525

26+
"google.golang.org/grpc/codes"
27+
"google.golang.org/grpc/status"
28+
2629
pb "github.com/dapr/dapr/pkg/proto/runtime/v1"
2730
"github.com/dapr/go-sdk/service/common"
2831
)
@@ -37,10 +40,15 @@ type SubscriptionOptions struct {
3740
}
3841

3942
type Subscription struct {
43+
ctx context.Context
4044
stream pb.Dapr_SubscribeTopicEventsAlpha1Client
45+
4146
// lock locks concurrent writes to subscription stream.
4247
lock sync.Mutex
4348
closed atomic.Bool
49+
50+
createStream func(ctx context.Context, opts SubscriptionOptions) (pb.Dapr_SubscribeTopicEventsAlpha1Client, error)
51+
opts SubscriptionOptions
4452
}
4553

4654
type SubscriptionMessage struct {
@@ -55,7 +63,10 @@ func (c *GRPCClient) Subscribe(ctx context.Context, opts SubscriptionOptions) (*
5563
}
5664

5765
s := &Subscription{
58-
stream: stream,
66+
ctx: ctx,
67+
stream: stream,
68+
createStream: c.subscribeInitialRequest,
69+
opts: opts,
5970
}
6071

6172
return s, nil
@@ -101,52 +112,96 @@ func (s *Subscription) Close() error {
101112
}
102113

103114
func (s *Subscription) Receive() (*SubscriptionMessage, error) {
104-
resp, err := s.stream.Recv()
105-
if err != nil {
106-
return nil, err
107-
}
108-
event := resp.GetEventMessage()
109-
110-
data := any(event.GetData())
111-
if len(event.GetData()) > 0 {
112-
mediaType, _, err := mime.ParseMediaType(event.GetDataContentType())
113-
if err == nil {
114-
var v interface{}
115-
switch mediaType {
116-
case "application/json":
117-
if err := json.Unmarshal(event.GetData(), &v); err == nil {
118-
data = v
115+
for {
116+
resp, err := s.stream.Recv()
117+
if err != nil {
118+
select {
119+
case <-s.ctx.Done():
120+
return nil, errors.New("subscription context closed")
121+
default:
122+
// proceed to check the gRPC status error
123+
}
124+
125+
st, ok := status.FromError(err)
126+
if !ok {
127+
// not a grpc status error
128+
return nil, err
129+
}
130+
131+
switch st.Code() {
132+
case codes.Unavailable, codes.Unknown:
133+
logger.Printf("gRPC error while reading from stream: %s (code=%v)",
134+
st.Message(), st.Code())
135+
// close the current stream and reconnect
136+
if s.closed.Load() {
137+
return nil, errors.New("subscription is permanently closed; cannot reconnect")
138+
}
139+
if err := s.closeStreamOnly(); err != nil {
140+
logger.Printf("error closing current stream: %v", err)
119141
}
120-
case "text/plain":
121-
// Assume UTF-8 encoded string.
122-
data = string(event.GetData())
142+
143+
newStream, nerr := s.createStream(s.ctx, s.opts)
144+
if nerr != nil {
145+
return nil, errors.New("re-subscribe failed")
146+
}
147+
148+
s.lock.Lock()
149+
s.stream = newStream
150+
s.lock.Unlock()
151+
152+
// try receiving again
153+
continue
154+
155+
case codes.Canceled:
156+
return nil, errors.New("stream canceled")
157+
123158
default:
124-
if strings.HasPrefix(mediaType, "application/") &&
125-
strings.HasSuffix(mediaType, "+json") {
159+
return nil, errors.New("subscription recv error")
160+
}
161+
}
162+
163+
event := resp.GetEventMessage()
164+
data := any(event.GetData())
165+
if len(event.GetData()) > 0 {
166+
mediaType, _, err := mime.ParseMediaType(event.GetDataContentType())
167+
if err == nil {
168+
var v interface{}
169+
switch mediaType {
170+
case "application/json":
126171
if err := json.Unmarshal(event.GetData(), &v); err == nil {
127172
data = v
128173
}
174+
case "text/plain":
175+
// Assume UTF-8 encoded string.
176+
data = string(event.GetData())
177+
default:
178+
if strings.HasPrefix(mediaType, "application/") &&
179+
strings.HasSuffix(mediaType, "+json") {
180+
if err := json.Unmarshal(event.GetData(), &v); err == nil {
181+
data = v
182+
}
183+
}
129184
}
130185
}
131186
}
132-
}
133187

134-
topicEvent := &common.TopicEvent{
135-
ID: event.GetId(),
136-
Source: event.GetSource(),
137-
Type: event.GetType(),
138-
SpecVersion: event.GetSpecVersion(),
139-
DataContentType: event.GetDataContentType(),
140-
Data: data,
141-
RawData: event.GetData(),
142-
Topic: event.GetTopic(),
143-
PubsubName: event.GetPubsubName(),
144-
}
188+
topicEvent := &common.TopicEvent{
189+
ID: event.GetId(),
190+
Source: event.GetSource(),
191+
Type: event.GetType(),
192+
SpecVersion: event.GetSpecVersion(),
193+
DataContentType: event.GetDataContentType(),
194+
Data: data,
195+
RawData: event.GetData(),
196+
Topic: event.GetTopic(),
197+
PubsubName: event.GetPubsubName(),
198+
}
145199

146-
return &SubscriptionMessage{
147-
sub: s,
148-
TopicEvent: topicEvent,
149-
}, nil
200+
return &SubscriptionMessage{
201+
sub: s,
202+
TopicEvent: topicEvent,
203+
}, nil
204+
}
150205
}
151206

152207
func (s *SubscriptionMessage) Success() error {
@@ -232,3 +287,13 @@ func (c *GRPCClient) subscribeInitialRequest(ctx context.Context, opts Subscript
232287

233288
return stream, nil
234289
}
290+
291+
func (s *Subscription) closeStreamOnly() error {
292+
s.lock.Lock()
293+
defer s.lock.Unlock()
294+
295+
if s.stream != nil {
296+
return s.stream.CloseSend()
297+
}
298+
return nil
299+
}

daprdocs/content/en/go-sdk-docs/go-client/_index.md

+85
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,91 @@ resp, err = client.InvokeMethodWithContent(ctx, "app-id", "method-name", "post",
8888

8989
For a full guide on service invocation, visit [How-To: Invoke a service]({{< ref howto-invoke-discover-services.md >}}).
9090

91+
### Workflows
92+
93+
Workflows and their activities can be authored and managed using the Dapr Go SDK like so:
94+
95+
```go
96+
import (
97+
...
98+
"github.com/dapr/go-sdk/workflow"
99+
...
100+
)
101+
102+
func ExampleWorkflow(ctx *workflow.WorkflowContext) (any, error) {
103+
var output string
104+
input := "world"
105+
106+
if err := ctx.CallActivity(ExampleActivity, workflow.ActivityInput(input)).Await(&output); err != nil {
107+
return nil, err
108+
}
109+
110+
// Print output - "hello world"
111+
fmt.Println(output)
112+
113+
return nil, nil
114+
}
115+
116+
func ExampleActivity(ctx workflow.ActivityContext) (any, error) {
117+
var input int
118+
if err := ctx.GetInput(&input); err != nil {
119+
return "", err
120+
}
121+
122+
return fmt.Sprintf("hello %s", input), nil
123+
}
124+
125+
func main() {
126+
// Create a workflow worker
127+
w, err := workflow.NewWorker()
128+
if err != nil {
129+
log.Fatalf("error creating worker: %v", err)
130+
}
131+
132+
// Register the workflow
133+
w.RegisterWorkflow(ExampleWorkflow)
134+
135+
// Register the activity
136+
w.RegisterActivity(ExampleActivity)
137+
138+
// Start workflow runner
139+
if err := w.Start(); err != nil {
140+
log.Fatal(err)
141+
}
142+
143+
// Create a workflow client
144+
wfClient, err := workflow.NewClient()
145+
if err != nil {
146+
log.Fatal(err)
147+
}
148+
149+
// Start a new workflow
150+
id, err := wfClient.ScheduleNewWorkflow(context.Background(), "ExampleWorkflow")
151+
if err != nil {
152+
log.Fatal(err)
153+
}
154+
155+
// Wait for the workflow to complete
156+
metadata, err := wfClient.WaitForWorkflowCompletion(ctx, id)
157+
if err != nil {
158+
log.Fatal(err)
159+
}
160+
161+
// Print workflow status post-completion
162+
fmt.Println(metadata.RuntimeStatus)
163+
164+
// Shutdown Worker
165+
w.Shutdown()
166+
}
167+
```
168+
169+
- For a more comprehensive guide on workflows visit these How-To guides:
170+
- [How-To: Author a workflow]({{< ref howto-author-workflow.md >}}).
171+
- [How-To: Manage a workflow]({{< ref howto-manage-workflow.md >}}).
172+
- Visit the Go SDK Examples to jump into complete examples:
173+
- [Workflow Example](https://github.com/dapr/go-sdk/tree/main/examples/workflow)
174+
- [Workflow - Parallelised](https://github.com/dapr/go-sdk/tree/main/examples/workflow-parallel)
175+
91176
### State Management
92177

93178
For simple use-cases, Dapr client provides easy to use `Save`, `Get`, `Delete` methods:

examples/conversation/main.go

+4-3
Original file line numberDiff line numberDiff line change
@@ -17,8 +17,9 @@ package main
1717
import (
1818
"context"
1919
"fmt"
20-
dapr "github.com/dapr/go-sdk/client"
2120
"log"
21+
22+
dapr "github.com/dapr/go-sdk/client"
2223
)
2324

2425
func main() {
@@ -28,12 +29,12 @@ func main() {
2829
}
2930

3031
input := dapr.ConversationInput{
31-
Message: "hello world",
32+
Content: "hello world",
3233
// Role: nil, // Optional
3334
// ScrubPII: nil, // Optional
3435
}
3536

36-
fmt.Printf("conversation input: %s\n", input.Message)
37+
fmt.Printf("conversation input: %s\n", input.Content)
3738

3839
var conversationComponent = "echo"
3940

examples/go.mod

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
module github.com/dapr/go-sdk/examples
22

3-
go 1.23.5
3+
go 1.23.6
44

55
replace github.com/dapr/go-sdk => ../
66

@@ -18,9 +18,9 @@ require (
1818
github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 // indirect
1919
github.com/cenkalti/backoff/v4 v4.3.0 // indirect
2020
github.com/cespare/xxhash/v2 v2.3.0 // indirect
21-
github.com/dapr/dapr v1.15.0-rc.9 // indirect
22-
github.com/dapr/durabletask-go v0.5.1-0.20250124181508-a1b42ae65aee // indirect
23-
github.com/dapr/kit v0.13.1-0.20250129050741-c46009f360b0 // indirect
21+
github.com/dapr/dapr v1.15.0-rc.17 // indirect
22+
github.com/dapr/durabletask-go v0.6.3 // indirect
23+
github.com/dapr/kit v0.15.0 // indirect
2424
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
2525
github.com/go-chi/chi/v5 v5.1.0 // indirect
2626
github.com/go-logr/logr v1.4.2 // indirect

0 commit comments

Comments
 (0)