Swift Firebase changing value is unstable - ios

I'm trying to change the value in Firebase, using TableView.
And Here is the Swift code in didSelectRowAt.
if let result = snapshot.children.allObjects as? [DataSnapshot] {
for child in result {
let nameDB = child.childSnapshot(forPath: "contactName").value! as! String
if nameDB == receiverName {
let autoID = child.key as String
if receiverAmount.characters.contains("-") {
self.dbRef.child("user/contacts/\(autoID)/borrowAmount").setValue("")
self.dbRef.child("user/contacts/\(autoID)/borrowDueDate").setValue("")
}
else {
self.dbRef.child("user/contacts/\(autoID)/lendAmount").setValue("")
self.dbRef.child("user/contacts/\(autoID)/lendDueDate").setValue("2099-12-31")
}
But While my app is, Values are unstable.
Original value and new value are endlessly changed, and then my app is stopped.
Any idea for solving this problem...?
Thank you!!

1) As you are using a tableView I am sure you are having al values required even if you do not use for loop ? instead using for loop on didselectAction just perform value update directly by creating a reference not using for loop.
2)Due to for loop Condition you are getting unstable outputs So, Just avoid for loop here and Directly update values

Related

Swift: If I remove a Value, why removed Firebase this Value again if I insert it again?

If I remove a value and try to insert it again, firebase remove this value automatically and I can't insert this value. Why is that and how can I break this issue?
ref.child("Matches").child(self.replayUserDepartment).child(DatingController.finalUserID).observe(.value) { (snapshot) in
if snapshot.exists(){
for child in snapshot.children{
let snap = child as! DataSnapshot
if (snap.value as? String) == self.replayUserUid{
ref.child("Matches").child(self.replayUserDepartment).child(DatingController.finalUserID).child(snap.key).setValue(nil)
return
}
}
}
}
In this Code I set a value that exists to nil because I want to delete this value. After I do this I want to insert the same value that I have removed before. If I do that, Firebase goes there and delete this value automatically without my agreement. And I cannot insert this specific value again.

Firebase setValue method is taking time

I am using firebase in my iOs app.
When I use the setValue method for updating node value, it takes around 5-10 secs to update value.
How can I overcome this delay?
Code
let shiftReportsRef = Database.database().reference().child(Constants.FireBase().users).child(Constants.FireBase().shiftReports).child(firebaseToken).child(cashierId.stringValue).child(todayDate).child(posPointId.stringValue).child(shiftid.stringValue)
shiftReportsRef?.observe(.value, with: { (peakHoursSnapShot) in
self.shiftDetails = peakHoursSnapShot.value as? [String: AnyObject] ?? [:]
let lastUpdated = Date().gmtDateTime(.dateTimeSec)
self.shiftDetails!["lastUpdated"] = lastUpdated as AnyObject
if self.hasUpdatedValues == false
{
self.updatePeakHoursAndLastUpdateDate(amount, refndedAmount, pymtType, timeSlt)
}
self.hasUpdatedValues = true
})
The code in the question is a little unclear as to what the expected outcome is. Let's walk through it...
First shiftReportRef is defined
let shiftReportsRef =
Database.database().reference().child(Constants.FireBase().users).child(Constants.FireBase().shiftReports).child(firebaseToken).child(cashierId.stringValue).child(todayDate).child(posPointId.stringValue).child(shiftid.stringValue)
then an an observer is added to that ref. The observer is .value which will observe any changes to that node and return them in a snapshot via an event
shiftReportsRef?.observe(.value
But then when the first event occurs and the code within the closure executes, the data at that location is overwritten
shiftReportsRef?.setValue(self.shiftDetails)
which then causes another .value event to fire, therefore overwriting the node again (I think you can see where this is going). Over and over.
So this isn't exactly an answer because we don't know the expected outcome but that's probably why there are delays in the app.

Issues reading data from Firebase Database

Okay I am reading from a database and when I print the individual variables they print out correctly. However it seems like the data refuses to append to the array. Anyone know why? I can't figure it out at all.
let commuteBuilder = Commutes()
Database.database().reference().child("Users").child(user).child("Trips").observe(DataEventType.childAdded, with: { (snapshot) in
//print(snapshot)
if let dict = snapshot.value as? NSDictionary {
commuteBuilder.distance = dict["Distance"] as! Double
commuteBuilder.title = dict["TripName"] as! String
commuteBuilder.transportType = (dict["Transport"] as? String)!
}
commuteArray.append(commuteBuilder)
})
print("helper")
print(commuteArray.count)
return commuteArray
The data is correctly added to the array, just not at the time that you print the array's contents.
If you change the code like this, you can see this:
let commuteBuilder = Commutes()
Database.database().reference().child("Users").child(user).child("Trips").observe(DataEventType.childAdded, with: { (snapshot) in
if let dict = snapshot.value as? NSDictionary {
commuteBuilder.distance = dict["Distance"] as! Double
commuteBuilder.title = dict["TripName"] as! String
commuteBuilder.transportType = (dict["Transport"] as? String)!
}
commuteArray.append(commuteBuilder)
print("added one, now have \(commuteArray.count)")
})
print("returning \(commuteArray.count)")
return commuteArray
You'll see it print something like this:
returning 0
added one, now have 1
added one, now have 2
etc.
This is likely not the output you expected. But it is working as intended. Firebase loads data from its database asynchronously. Instead of blocking your code, it lets the thread continue (so the user can continue using the app) and instead calls back to the code block you passed to observe when new data is available.
This means that by the time this code returns the array it is still empty, but it later adds items as they come in. This means that you cannot return data from a function in the way you are trying.
I find it easiest to change my way of thinking about code. Instead of "First get the data, then print it", I frame it as "Start getting the data. When data comes back, print it".
In the code above, I did this by moving the code that prints the count into the callback block. Instead of doing this, you can also create your own callback, which is called a completion handler or closure in Swift. You can find examples in this article, this article, this question Callback function syntax in Swift or of course in Apple's documentation.

How to Update uitableview from CloudKit using discoverAllIdentities

So, I am new to cloudKit and to working with multiple threads in general, which I think is the source of the problem here, so if I simply need to research more, please just comment so and I will take that to heart.
Here is my question:
I am working in Swift 3 Xcode 8.1
I have in my view controller this variable:
var contactsNearby: [String:CLLocation]?
Then at the end of ViewDidLoad I call one of my view controllers methods let's call it:
populateContactsNearby()
inside that method I call:
container.discoverAllIdentities(completionHandler: { (identities, error) in
for userIdentity in identities! {
self.container.publicCloudDatabase.fetch(withRecordID: userIdentity.userRecordID!, completionHandler: { (userRecord, error) in
let contactsLocation = userRecord?.object(forKey: "currentLocation")
if self.closeEnough(self.myLocation!, contactLocation: contactsLocation as! CLLocation) {
var contactsName = ""
contactsFirstName = userIdentity.nameComponents?.givenName
if contactsName != "" && contactsLocation != nil {
self.contactsNearby?["\(contactsName)"] = contactsLocation as? CLLocation
}
}
})
}
})
}
I apologize if I am missing or have an extra bracket somewhere. I have omitted some error checking code and so forth in order to get this down to bare-bones. So the goal of all that is to populate my contactsNearby Dictionary with data from CloudKit. A name as the key a location as the value. I want to use that data to populate a tableview. In the above code, the call to closeEnough is a call to another one of my view controllers methods to check if the contact from CloudKit has a location close enough to my user to be relevant to the apps purposes. Also myLocation is a variable that is populated before the segue. It holds the CLLocation of the app users current location.
The Problem:
The if statement:
if contactsName != "" && contactsLocation != nil { }
Appears to succeed. But my view controllers variable:
var contactsNearby: [String:CLLocation]?
Is never populated and I know there is data available in cloudKit.
If it's relevant here is some test code that I have in cellForRowAtIndexPath right now:
let contact = self.contactsNearby?.popFirst()
let name = contact?.key
if name != nil {
cell.textLabel?.text = name
}else {
cell.textLabel?.text = "nothing was there"
}
My rows alway populate with "nothing was there". I have seen answers where people have done CKQueries to update the UI, but in those answers, the user built the query themselves. That seems different from using a CloudKit function like discoverAllIdentities.
I have tried to be as specific as possible in asking this question. If this question could be improved please let me know. I think it's a question that could benefit the community.
Okay, I need to do some more testing, but I think I got it working. Thank you Paulw11 for your comment. It got me on the right track.
As it turns out there were 2 problems.
First, as pointed out I have an asynchronous call inside a for loop. As recommended, I used a dispatchGroup.
Inside the cloudKit call to discoverAllIdentities I declared a dispatchGroup, kind of like so:
var icloudDispatchGroup = DispatchGroup()
Then just inside the for loop that is going to make an async call, I enter the dispatchGroup:
icloudDispatchGroup.enter()
Then just before the end of the publicCloudDatabase.fetch completion handler I call:
icloudDispatchGroup.leave()
and
icloudDispatchGroup.wait()
Which, I believe, I'm still new to this remember, ends the dispatchGroup and causes the current thread to wait until that dispatchGroup finishes before allowing the current thread to continue.
The Above took care of the multithreading issue, but my contactsNearby[String:CLLocation]? Dictionary was still not being populated.
Which leads me to the 2nd problem
At the top of my view controller I declared my Dictionary:
var contactsNearby: [String: CLLocation]?
This declared a dictionary, but does not initialize it, which I don't think I fully realized, so when I attempted to populate it:
self.contactsNearby?["\(contactsName)"] = contactsLocation as? CLLocation
It quietly failed because it is optional and returned nil
So, in viewDidLoad before I even call populateContactsNearby I initialize the dictionary:
contactsNearby = [String:CLLocation]()
This does not make it cease to be an optional, which Swift being strongly typed would not allow, but merely initializes contactsNearby as an optional empty Dictionary.
At least, that is my understanding of what is going on. If anyone has a more elegant solution, I am always trying to improve.
In case you are wondering how I then update the UI, I do so with a property observer on the contactsNearby Dictionary. So the declaration of the dictionary at the top of the view controller looks like this:
var contactsNearby: [String: CLLocation]? {
didSet {
if (contactsNearby?.isEmpty)! || contactsNearby == nil {
return
}else{
DispatchQueue.main.sync {
self.nearbyTableView.reloadData()
}
}
}
}
I suppose I didn't really need to check for empty and nil. So then in cellForRowAtIndexPath I have something kind of like so:
let cell = tableview.dequeueReusableCell(withIdentifier: "nearbyCell", for: indexPath)
if contactsNearby?.isEmpty == false {
let contact = contactsNearby?.popFirst()
cell.textLabel?.text = contact?.key
}else {
cell.textLabel?.text = "Some Placeholder Text Here"
}
return cell
If anyone sees an error in my thinking or sees any of this heading for disaster, feel free to let me know. I still have a lot of testing to do, but I wanted to get back here and let you know what I have found.

Receiving Data using Firebase swift

I am trying to retrieve Childs from The Firebase Database using this code (which is declared in viewdidload)
let rootRef = FIRDatabase.database().reference().child("UsersInfo").child((FIRAuth.auth()?.currentUser?.uid)!)
rootRef.observeEventType(FIRDataEventType.Value, withBlock: { (snapshot) in
let hello = snapshot.value as! [String : AnyObject]
let usernamerecieved = hello["Username"] as! String
let Emailrecieved = hello["Email"] as! String
let bloodtyperecieved = hello["BloodType"] as! String
globalusername = usernamerecieved (EDITED)
})
and I've declared a global variable as such
var globalusername = "user"
Im trying to extract the usernamerecieved variable and casting it on a Global Variable. However when I print the global variable later on in the viewdidLoad I still get the initial value of the globalusername which is "user".
In this line of code : usernamerecieved = globalusername what you are doing in this line is that you are setting the value of the usernamerecieved to the value of globalusername i.e. 'user'
It should be globalusername = usernamerecieved.
Also sometimes it takes time to retrieve data from the server reasons could be numerous - slow network connection, heavy data retrieving such as images etc. but before it could be completed (Retrieving Action From The Server) it might be possible that line in which you are printing the globalusername gets called first , even before the completionBlock off the observeEventType is even completed (since its asynchronous , somewhat similar to dispatch_async).To tackle that you should call the printing call for globalusername inside the completionBlock or observe any change in it's value by using addObserver to the globalusername,which will let you know whenever the value of globalusername is changed or updated so that you can act accordingly .
observeEventType's block is asynchronous, it is possible that by the time you read the globalusername var, the block has not been completed yet.
Try printing globalusername inside the block and see the value

Resources