forked from TrueCloudLab/neoneo-go
rpcclient: refactor event-based waiting loop
Avoid receiver channels locks.
This commit is contained in:
parent
95e23c8e46
commit
1399496dfb
1 changed files with 78 additions and 75 deletions
|
@ -219,25 +219,80 @@ func (w *EventWaiter) Wait(h util.Uint256, vub uint32, err error) (res *state.Ap
|
||||||
func (w *EventWaiter) WaitAny(ctx context.Context, vub uint32, hashes ...util.Uint256) (res *state.AppExecResult, waitErr error) {
|
func (w *EventWaiter) WaitAny(ctx context.Context, vub uint32, hashes ...util.Uint256) (res *state.AppExecResult, waitErr error) {
|
||||||
var (
|
var (
|
||||||
wsWaitErr error
|
wsWaitErr error
|
||||||
bRcvr = make(chan *block.Block)
|
|
||||||
aerRcvr = make(chan *state.AppExecResult)
|
|
||||||
unsubErrs = make(chan error)
|
|
||||||
waitersActive int
|
waitersActive int
|
||||||
|
bRcvr = make(chan *block.Block, 2)
|
||||||
|
aerRcvr = make(chan *state.AppExecResult, len(hashes))
|
||||||
|
unsubErrs = make(chan error)
|
||||||
|
exit = make(chan struct{})
|
||||||
)
|
)
|
||||||
|
|
||||||
// Rollback to a poll-based waiter if needed.
|
// Execution event preceded the block event, thus wait until the VUB-th block to be sure.
|
||||||
defer func() {
|
since := vub
|
||||||
if wsWaitErr != nil {
|
blocksID, err := w.ws.ReceiveBlocks(&neorpc.BlockFilter{Since: &since}, bRcvr)
|
||||||
res, waitErr = w.polling.WaitAny(ctx, vub, hashes...)
|
if err != nil {
|
||||||
if waitErr != nil {
|
wsWaitErr = fmt.Errorf("failed to subscribe for new blocks: %w", err)
|
||||||
// Wrap the poll-based error, it's more important.
|
} else {
|
||||||
waitErr = fmt.Errorf("event-based error: %v; poll-based waiter error: %w", wsWaitErr, waitErr)
|
waitersActive++
|
||||||
|
go func() {
|
||||||
|
<-exit
|
||||||
|
err = w.ws.Unsubscribe(blocksID)
|
||||||
|
if err != nil {
|
||||||
|
unsubErrs <- fmt.Errorf("failed to unsubscribe from blocks (id: %s): %w", blocksID, err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
unsubErrs <- nil
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
if wsWaitErr == nil {
|
||||||
|
for _, h := range hashes {
|
||||||
|
txsID, err := w.ws.ReceiveExecutions(&neorpc.ExecutionFilter{Container: &h}, aerRcvr)
|
||||||
|
if err != nil {
|
||||||
|
wsWaitErr = fmt.Errorf("failed to subscribe for execution results: %w", err)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
waitersActive++
|
||||||
|
go func() {
|
||||||
|
<-exit
|
||||||
|
err = w.ws.Unsubscribe(txsID)
|
||||||
|
if err != nil {
|
||||||
|
unsubErrs <- fmt.Errorf("failed to unsubscribe from transactions (id: %s): %w", txsID, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
unsubErrs <- nil
|
||||||
|
}()
|
||||||
}
|
}
|
||||||
}()
|
}
|
||||||
|
|
||||||
// Drain receivers to avoid other notification receivers blocking.
|
if wsWaitErr == nil {
|
||||||
defer func() {
|
select {
|
||||||
|
case b, ok := <-bRcvr:
|
||||||
|
if !ok {
|
||||||
|
// We're toast, retry with non-ws client.
|
||||||
|
wsWaitErr = ErrMissedEvent
|
||||||
|
break
|
||||||
|
}
|
||||||
|
// We can easily end up in a situation when subscription was performed too late and
|
||||||
|
// the desired transaction and VUB-th block have already got accepted before the
|
||||||
|
// subscription happened. Thus, always retry with non-ws client, it will perform
|
||||||
|
// AER requests and make sure.
|
||||||
|
wsWaitErr = fmt.Errorf("block #%d was received by EventWaiter", b.Index)
|
||||||
|
case aer, ok := <-aerRcvr:
|
||||||
|
if !ok {
|
||||||
|
// We're toast, retry with non-ws client.
|
||||||
|
wsWaitErr = ErrMissedEvent
|
||||||
|
break
|
||||||
|
}
|
||||||
|
res = aer
|
||||||
|
case <-w.ws.Context().Done():
|
||||||
|
waitErr = fmt.Errorf("%w: %v", ErrContextDone, w.ws.Context().Err())
|
||||||
|
case <-ctx.Done():
|
||||||
|
waitErr = fmt.Errorf("%w: %v", ErrContextDone, ctx.Err())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
close(exit)
|
||||||
|
|
||||||
|
if waitersActive > 0 {
|
||||||
|
// Drain receivers to avoid other notification receivers blocking.
|
||||||
drainLoop:
|
drainLoop:
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
|
@ -260,72 +315,20 @@ func (w *EventWaiter) WaitAny(ctx context.Context, vub uint32, hashes ...util.Ui
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if wsWaitErr == nil || !errors.Is(wsWaitErr, ErrMissedEvent) {
|
|
||||||
close(bRcvr)
|
|
||||||
close(aerRcvr)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
// Execution event preceded the block event, thus wait until the VUB-th block to be sure.
|
|
||||||
since := vub
|
|
||||||
blocksID, err := w.ws.ReceiveBlocks(&neorpc.BlockFilter{Since: &since}, bRcvr)
|
|
||||||
if err != nil {
|
|
||||||
wsWaitErr = fmt.Errorf("failed to subscribe for new blocks: %w", err)
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
waitersActive++
|
if wsWaitErr == nil || !errors.Is(wsWaitErr, ErrMissedEvent) {
|
||||||
defer func() {
|
close(bRcvr)
|
||||||
go func() {
|
close(aerRcvr)
|
||||||
err = w.ws.Unsubscribe(blocksID)
|
|
||||||
if err != nil {
|
|
||||||
unsubErrs <- fmt.Errorf("failed to unsubscribe from blocks (id: %s): %w", blocksID, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
unsubErrs <- nil
|
|
||||||
}()
|
|
||||||
}()
|
|
||||||
for _, h := range hashes {
|
|
||||||
txsID, err := w.ws.ReceiveExecutions(&neorpc.ExecutionFilter{Container: &h}, aerRcvr)
|
|
||||||
if err != nil {
|
|
||||||
wsWaitErr = fmt.Errorf("failed to subscribe for execution results: %w", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
waitersActive++
|
|
||||||
defer func() {
|
|
||||||
go func() {
|
|
||||||
err = w.ws.Unsubscribe(txsID)
|
|
||||||
if err != nil {
|
|
||||||
unsubErrs <- fmt.Errorf("failed to unsubscribe from transactions (id: %s): %w", txsID, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
unsubErrs <- nil
|
|
||||||
}()
|
|
||||||
}()
|
|
||||||
}
|
}
|
||||||
|
close(unsubErrs)
|
||||||
|
|
||||||
select {
|
// Rollback to a poll-based waiter if needed.
|
||||||
case b, ok := <-bRcvr:
|
if wsWaitErr != nil && waitErr == nil {
|
||||||
if !ok {
|
res, waitErr = w.polling.WaitAny(ctx, vub, hashes...)
|
||||||
// We're toast, retry with non-ws client.
|
if waitErr != nil {
|
||||||
wsWaitErr = ErrMissedEvent
|
// Wrap the poll-based error, it's more important.
|
||||||
return
|
waitErr = fmt.Errorf("event-based error: %v; poll-based waiter error: %w", wsWaitErr, waitErr)
|
||||||
}
|
}
|
||||||
// We can easily end up in a situation when subscription was performed too late and
|
|
||||||
// the desired transaction and VUB-th block have already got accepted before the
|
|
||||||
// subscription happened. Thus, always retry with non-ws client, it will perform
|
|
||||||
// AER requests and make sure.
|
|
||||||
wsWaitErr = fmt.Errorf("block #%d was received by EventWaiter", b.Index)
|
|
||||||
case aer, ok := <-aerRcvr:
|
|
||||||
if !ok {
|
|
||||||
// We're toast, retry with non-ws client.
|
|
||||||
wsWaitErr = ErrMissedEvent
|
|
||||||
return
|
|
||||||
}
|
|
||||||
res = aer
|
|
||||||
case <-w.ws.Context().Done():
|
|
||||||
waitErr = fmt.Errorf("%w: %v", ErrContextDone, w.ws.Context().Err())
|
|
||||||
case <-ctx.Done():
|
|
||||||
waitErr = fmt.Errorf("%w: %v", ErrContextDone, ctx.Err())
|
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue