Search code examples
dartdart-async

Returned future is always 1 less than the expected value


I have the following dart code:

.dart

Future<int> currentMrn( ) async {
  var rootRef = await firebaseClient;
  var mrnRef = await rootRef.child( 'ids/mrn' );
  var event = await mrnRef.onValue.first;
  DataSnapshot ss = event.snapshot;

  return ss.val( );
}

Future<int> nextMrn( ) async {
  int curMrn = 0;
  var rootRef = await firebaseClient;
  Future<int> futureMrn = currentMrn( );

  futureMrn.then( ( int value ) {
    if ( value == null ) {
      rootRef.child('ids/mrn').set(curMrn);
      //futureMrn = Future.value([curMrn]);
    }
    else {
      curMrn = value + 1;
      rootRef.child('ids/mrn').set(curMrn);
      //futureMrn = Future.value([curMrn]);
    }
  } );

  return currentMrn( );
}

The code is called as such:

.dart

  nextMrn().then((int value) {
    print(value);
  });

However, the printed value is ALWAYS 1 less than the value in firebase.

It seems that currentMrn() is not getting the new updated value.

I am using the dart firebase package at https://pub.dartlang.org/packages/firebase

Thanks for your help.

EDIT 1

Running the future.then (first version) of nextMrn() throws the following exception

Exception: Uncaught Error: The null object does not have a method 'cancel'.

NoSuchMethodError: method not found: 'cancel'
Receiver: null
Arguments: []
Stack Trace:
#0      Object._noSuchMethod (dart:core-patch/object_patch.dart:42)
#1      Object.noSuchMethod (dart:core-patch/object_patch.dart:45)
#2      _cancelAndValue (dart:async/stream_pipe.dart:58)
#3      Stream.first.<anonymous closure> (dart:async/stream.dart:937)
#4      _RootZone.runUnaryGuarded (dart:async/zone.dart:1104)
#5      _BufferingStreamSubscription._sendData (dart:async/stream_impl.dart:341)
#6      _BufferingStreamSubscription._add (dart:async/stream_impl.dart:270)
#7      _SyncBroadcastStreamController._sendData (dart:async/broadcast_stream_controller.dart:362)
#8      _BroadcastStreamController.add (dart:async/broadcast_stream_controller.dart:237)
#9      Query._createStream.addEvent (package:firebase/src/firebase.dart:531:12)
#10     JsObject._callMethod (dart:js:678)
#11     JsObject.callMethod (dart:js:618)
#12     Query._createStream.startListen (package:firebase/src/firebase.dart:539:11)
#13     _runGuarded (dart:async/stream_controller.dart:769)
#14     _BroadcastStreamController._subscribe (dart:async/broadcast_stream_controller.dart:199)
#15     _ControllerStream._createSubscription (dart:async/stream_controller.dart:787)
#16     _StreamImpl.listen (dart:async/stream_impl.dart:474)
#17     Stream.first (dart:async/stream.dart:935)
#18     currentMrn.<currentMrn_async_body> (package:epimss_shared/src/epimss_shared_db_client.dart:34:36)
#19     _RootZone.runUnary (dart:async/zone.dart:1166)
#20     _Future._propagateToListeners.handleValueCallback (dart:async/future_impl.dart:494)
#21     _Future._propagateToListeners (dart:async/future_impl.dart:577)
#22     _Future._completeWithValue (dart:async/future_impl.dart:368)
#23     _Future._asyncComplete.<anonymous closure> (dart:async/future_impl.dart:422)
#24     _microtaskLoop (dart:async/schedule_microtask.dart:43)
#25     _microtaskLoopEntry (dart:async/schedule_microtask.dart:52)
#26     _ScheduleImmediateHelper._handleMutation (dart:html:42529)

Line #18 above refers to the

var event = await mrnRef.onValue.first;

line in the currentMrn() method. Seems like an attempt is made to cancel an event. Not certain.

EDIT 2

Because all prior attempts returns an integer 1 less than is expected, I have included the code for currentMrn() directly in nextMrn() as follows:

Future<int> currentMrn( ) async {
  var rootRef = await firebaseClient;
  var mrnRef = await rootRef.child( 'ids/mrn' );
  var event = await mrnRef.onValue.first;
  DataSnapshot ss = event.snapshot;

  return ss.val( );
}

Future<int> nextMrn( ) async {
  var curMrn = 0;

  var rootRef = await firebaseClient;
  var mrnRef = await rootRef.child( 'ids/mrn' );
  var event = await mrnRef.onValue.first;
  DataSnapshot ss = event.snapshot;

  var value = ss.val( );

  if ( value == null ) {
    curMrn += curMrn + 1;
    rootRef.child( 'ids/mrn' ).set( curMrn );
  }
  else {
    curMrn = value + 1;
    rootRef.child( 'ids/mrn' ).set( curMrn );
  }
  return curMrn;
  //TODO correct Bad state: Cannot fire new event. Controller is already firing an event
}

This correctly returns the expected value when nextMrn() is invoked. However, it throws the exception below:

FIREBASE WARNING: Exception was thrown by user callback.  
Uncaught Unhandled exception:
Bad state: Cannot fire new event. Controller is already firing an event
#0      _BroadcastStreamController.add (dart:async/broadcast_stream_controller.dart:236)
#1      Query._createStream.addEvent (package:firebase/src/firebase.dart:531:12)
#2      JsObject._callMethod (dart:js:678)
#3      JsObject.callMethod (dart:js:618)
#4      Firebase.set (package:firebase/src/firebase.dart:258:9)
#5      nextMrn.<nextMrn_async_body> (package:epimss_shared/src/epimss_shared_db_client.dart:56:38)
#6      _RootZone.runUnary (dart:async/zone.dart:1166)
#7      _Future._propagateToListeners.handleValueCallback (dart:async/future_impl.dart:494)
#8      _Future._propagateToListeners (dart:async/future_impl.dart:577)
#9      _Future._complete (dart:async/future_impl.dart:358)
#10     _cancelAndValue (dart:async/stream_pipe.dart:62)
#11     Stream.first.<anonymous closure> (dart:async/stream.dart:937)
#12     _RootZone.runUnaryGuarded (dart:async/zone.dart:1104)
#13     _BufferingStreamSubscription._sendData (dart:async/stream_impl.dart:341)
#14     _BufferingStreamSubscription._add (dart:async/stream_impl.dart:270)
#15     _SyncBroadcastStreamController._sendData (dart:async/broadcast_stream_controller.dart:362)
#16     _BroadcastStreamController.add (dart:async/broadcast_stream_controller.dart:237)
#17     Query._createStream.addEvent (package:firebase/src/firebase.dart:531:12)
  (anonymous function)

that points to the following line in the refactored nextMrn()

rootRef.child( 'ids/mrn' ).set( curMrn );

Two questions: 1. How is this corrected. 2. How do I catch an error in the nextMrn()?

Thanks


Solution

  • Future<int> nextMrn( ) async {
      int curMrn = 0;
      var rootRef = await firebaseClient;
      Future<int> futureMrn = currentMrn( );
    
      // missing return leads to broken future chain
      return futureMrn.then( ( int value ) {
        if ( value == null ) {
          rootRef.child('ids/mrn').set(curMrn);
          //futureMrn = Future.value([curMrn]);
        }
        else {
          curMrn = value + 1;
          rootRef.child('ids/mrn').set(curMrn);
          //futureMrn = Future.value([curMrn]);
        }
      // add the actual return in the chain to ensure
      // id doesn't return before the calculation is completed
      }).then((_) => currentMrn( );
    }
    

    or with async/await

    Future<int> nextMrn( ) async {
      int curMrn = 0;
      var rootRef = await firebaseClient;
      var futureMrn = currentMrn( );
      var value = await futureMrn;
      if ( value == null ) {
        rootRef.child('ids/mrn').set(curMrn);
        //futureMrn = Future.value([curMrn]);
      }
      else {
        curMrn = value + 1;
        rootRef.child('ids/mrn').set(curMrn);
        //futureMrn = Future.value([curMrn]);
      }
      return currentMrn( );
    }
    

    I couldn't really figure out what exactly the intention of your code is. Hope it works anyway.