I have the following dart code:
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:
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.
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)
var event = await mrnRef.onValue.first;
line in the currentMrn() method. Seems like an attempt is made to cancel an event. Not certain.
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
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.