Skip to content

Commit

Permalink
Merge pull request #3271 from oasisprotocol/kostko/fix/rt-client-shut…
Browse files Browse the repository at this point in the history
…down

go/runtime/client: Fix possible panic on shutdown
  • Loading branch information
kostko authored Sep 14, 2020
2 parents 570e3ad + 1d217af commit c200555
Show file tree
Hide file tree
Showing 3 changed files with 27 additions and 15 deletions.
1 change: 1 addition & 0 deletions .changelog/3271.bugfix.md
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
go/runtime/client: Fix possible panic on shutdown
34 changes: 23 additions & 11 deletions go/runtime/client/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -84,37 +84,49 @@ func (c *runtimeClient) SubmitTx(ctx context.Context, request *api.SubmitTxReque
}
c.Unlock()

// Send a request for watching a new runtime transaction.
respCh := make(chan *watchResult)
var requestID hash.Hash
requestID.FromBytes(request.Data)
watcher.newCh <- &watchRequest{
id: &requestID,
req := &watchRequest{
ctx: ctx,
respCh: respCh,
}
req.id.FromBytes(request.Data)
select {
case <-ctx.Done():
// The context we're working in was canceled, abort.
return nil, ctx.Err()
case <-c.common.ctx.Done():
// Client is shutting down.
return nil, fmt.Errorf("client: shutting down")
case watcher.newCh <- req:
}

// Wait for response, handling retries if/when needed.
for {
var resp *watchResult
var ok bool

select {
case <-ctx.Done():
// The context we're working in was canceled, abort.
return nil, context.Canceled

return nil, ctx.Err()
case <-c.common.ctx.Done():
// Client is shutting down.
return nil, fmt.Errorf("client: shutting down")
case resp, ok = <-respCh:
// The main event is getting a response from the watcher, handled below.
if !ok {
return nil, fmt.Errorf("client: block watch channel closed unexpectedly (unknown error)")
}

// The main event is getting a response from the watcher, handled below. If there is
// no result yet, this means that we need to retry publish.
if resp.result == nil {
break
}

return resp.result, nil
}

if !ok {
return nil, fmt.Errorf("client: block watch channel closed unexpectedly (unknown error)")
}

c.common.p2p.Publish(context.Background(), request.RuntimeID, &p2p.Message{
Tx: &executor.Tx{
Data: request.Data,
Expand Down
7 changes: 3 additions & 4 deletions go/runtime/client/watcher.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ const (
)

type watchRequest struct {
id *hash.Hash
id hash.Hash
ctx context.Context
respCh chan *watchResult
height int64
Expand Down Expand Up @@ -96,7 +96,6 @@ func (w *blockWatcher) checkBlock(blk *block.Block) {

func (w *blockWatcher) watch() {
defer func() {
close(w.newCh)
for _, watch := range w.watched {
close(watch.respCh)
}
Expand Down Expand Up @@ -204,13 +203,13 @@ func (w *blockWatcher) watch() {
}

case newWatch := <-w.newCh:
w.watched[*newWatch.id] = newWatch
w.watched[newWatch.id] = newWatch

res := &watchResult{
groupVersion: latestGroupVersion,
}
if newWatch.send(res, latestHeight) != nil {
delete(w.watched, *newWatch.id)
delete(w.watched, newWatch.id)
}

case <-w.stopCh:
Expand Down

0 comments on commit c200555

Please sign in to comment.