Firebase Query snapshot nil?

Я создаю простое приложение для чата, чтобы узнать Swift и Firebase. У меня есть запрос, который проверяет, чтобы сообщения в чате были загружены в tableView. Запрос возвращает моментальный снимок, но у меня возникают проблемы с получением данных для их хранения и загрузки. Я создал словарь из моментального снимка, чтобы получить отдельные дочерние значения, но они возвращают нуль:

query.observeSingleEvent(of: .childAdded, with: { snapshot in

            print(snapshot)

            let indexPaths = self.messages.enumerated().map {
                IndexPath(row: $0.0, section: 0)
            }

            guard let messagesDict = snapshot.value as? [String:AnyObject] else { return }
            guard let sender = messagesDict["sender"] as? String else { return }
            guard let text = messagesDict["text"] as? String else { return }
            guard let timestamp = messagesDict["timestamp"] as? Int else { return }

            let message = Message.init(key: snapshot.key, sender: sender as! String, text: text as! String, timestamp: timestamp)

            self.messages.append(message)

            self.tableView.reloadData()
        })

po messagesDict возвращает данные в точке останова:

▿ 1 element
  ▿ 0 : 2 elements
    - key : "-KotqLUUucaRagTRt967"
    ▿ value : 3 elements
      ▿ 0 : 2 elements
        - key : sender
        - value : eGTYRSo81JefgasYLRHUFHUTnEC3
      ▿ 1 : 2 elements
        - key : text
        - value : test
      ▿ 2 : 2 elements
        - key : timestamp
        - value : 1499914135546

Приложение переходит к следующей точке останова, где я устанавливаю guard let sender, но приложение обращается к нему и останавливает выполнение остальной части кода. Как правильно получить доступ к элементам отправителя, текста и временных меток, чтобы я мог установить их в переменные и сохранить их в массиве для моей таблицы?

Любая помощь будет принята с благодарностью. Я застрял в этом какое-то время, и я мог бы использовать еще пару глаз!

Спасибо!!

Дополнительная информация:

Query:

let query = rootRef.child("chat-messages/\(chatKey)")

Структура базы данных:

{
  "chat-messages" : {
    "-KotqKI4zF9cS5XMR9WU" : {
      "-KotqLUUucaRagTRt967" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499914135546
      }
    },
    "-KotqvBoFFzOTY6Q1fO7" : {
      "-KotqwGAJ85NuAp-Lh3Z" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499914290279
      }
    },
    "-KotsNgF0ou5MIDn2l5K" : {
      "-KotsOSXw7OS_3xuXZgP" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499914671997
      },
      "-KotsPqrCpipMLulNNFl" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test44",
        "timestamp" : 1499914677715
      }
    },
    "-Kotsv1Vv19dEF2q7XQu" : {
      "-KotsvYO9JvLY8gdTnA0" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499914811675
      },
      "-KotsvxKXzIwbyvbwLPp" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499914813335
      },
      "-Kotsw9vLiEf6KquU__e" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499914814205
      },
      "-KotswHnzmfkEXih1O7y" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499914814709
      },
      "-KotszHKBt254jZnUMAl" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test46456",
        "timestamp" : 1499914826967
      }
    },
    "-KoyheS01YxWExSNxl_H" : {
      "-Koyhf3G-FPEDrxQOcpU" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499995746607
      }
    },
    "-Koyjc_aBaF1kFE-8Bll" : {
      "-Koyjd9PnO6sPOhS7P27" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499996263096
      },
      "-KoyjdzSEi5twTVQMGOu" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499996266491
      },
      "-Koyje2C9E524kV7nhLY" : {
        "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
        "text" : "test",
        "timestamp" : 1499996266730
      }
    },
    "-KpfUGw45F1FU2IrR87A" : {
      "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
      "text" : "test",
      "timestamp" : 1500746948396
    },
    "-KpfUHQGuTDi4yZutS5K" : {
      "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
      "text" : "test",
      "timestamp" : 1500746950368
    },
    "-KpfUHdSXeTzauATsDAP" : {
      "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
      "text" : "test",
      "timestamp" : 1500746951277
    },
    "-Kpfia0fXsoqpRtIr1kU" : {
      "sender" : "eGTYRSo81JefgasYLRHUFHUTnEC3",
      "text" : "gerg",
      "timestamp" : 1500750962815
    }
  },
  "chat-participants" : {
    "-KotqKI4zF9cS5XMR9WU" : {
      "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
        "username" : "tester"
      }
    },
    "-KotqvBoFFzOTY6Q1fO7" : {
      "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
        "username" : "tester"
      }
    },
    "-KotsNgF0ou5MIDn2l5K" : {
      "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
        "username" : "tester"
      }
    },
    "-Kotsv1Vv19dEF2q7XQu" : {
      "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
        "username" : "tester"
      }
    },
    "-KoyheS01YxWExSNxl_H" : {
      "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
        "username" : "tester"
      }
    },
    "-Koyjc_aBaF1kFE-8Bll" : {
      "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
        "username" : "tester"
      }
    },
    "-KpfLBjQdPiX2kSrl1yF" : {
      "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
        "username" : "tester"
      }
    }
  },
  "chats" : {
    "-KotqKI4zF9cS5XMR9WU" : {
      "currentName" : "test",
      "date" : 1499914130657
    },
    "-KotqvBoFFzOTY6Q1fO7" : {
      "currentName" : "test",
      "date" : 1499914285904
    },
    "-KotsNgF0ou5MIDn2l5K" : {
      "currentName" : "test",
      "date" : 1499914668844
    },
    "-Kotsv1Vv19dEF2q7XQu" : {
      "currentName" : "test",
      "date" : 1499914809570
    },
    "-KoyheS01YxWExSNxl_H" : {
      "currentName" : "test",
      "date" : 1499995744095
    },
    "-Koyjc_aBaF1kFE-8Bll" : {
      "currentName" : "test",
      "date" : 1499996260740
    },
    "-KpfLBjQdPiX2kSrl1yF" : {
      "currentName" : "test3222",
      "date" : 1500744567786
    }
  },
  "user-chats" : {
    "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
      "-KotqKI4zF9cS5XMR9WU" : {
        "chatKey" : "-KotqKI4zF9cS5XMR9WU",
        "currentName" : "test"
      },
      "-KotqvBoFFzOTY6Q1fO7" : {
        "chatKey" : "-KotqvBoFFzOTY6Q1fO7",
        "currentName" : "test"
      },
      "-KotsNgF0ou5MIDn2l5K" : {
        "chatKey" : "-KotsNgF0ou5MIDn2l5K",
        "currentName" : "test"
      },
      "-Kotsv1Vv19dEF2q7XQu" : {
        "chatKey" : "-Kotsv1Vv19dEF2q7XQu",
        "currentName" : "test"
      },
      "-KoyheS01YxWExSNxl_H" : {
        "chatKey" : "-KoyheS01YxWExSNxl_H",
        "currentName" : "test"
      },
      "-Koyjc_aBaF1kFE-8Bll" : {
        "chatKey" : "-Koyjc_aBaF1kFE-8Bll",
        "currentName" : "test"
      },
      "-KpfLBjQdPiX2kSrl1yF" : {
        "chatKey" : "-KpfLBjQdPiX2kSrl1yF",
        "currentName" : "test3222"
      }
    }
  },
  "user-friends" : {
    "36HDn5XyxjMCcMluJJ9KPBOCDMm1" : {
      "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
        "username" : "tester"
      }
    },
    "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
      "36HDn5XyxjMCcMluJJ9KPBOCDMm1" : {
        "username" : "tester2"
      }
    }
  },
  "user-invites" : {
    "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
      "36HDn5XyxjMCcMluJJ9KPBOCDMm1" : {
        "username" : "tester2"
      }
    }
  },
  "users" : {
    "36HDn5XyxjMCcMluJJ9KPBOCDMm1" : {
      "email" : "[email protected]",
      "username" : "tester2"
    },
    "eGTYRSo81JefgasYLRHUFHUTnEC3" : {
      "email" : "[email protected]",
      "username" : "tester"
    }
  }
}

Ответ 1

Я не думаю, что вы хотите обработчик ChildAdded, поэтому я иду с примером registerSingleEvent, так как вы хотите запросить данные из базы данных в то время без триггеров. При использовании функции observSingleEvent важно поддерживать синхронизацию базы данных. Я бы рекомендовал использовать следующий код:

query.keepSynched(true) //keeps data in sync with database, if you have data persistince on in your appDelegate
query.observeSingleEvent(of: .value, with: { (snapshot) in //notice the changed here
            print(snapshot)
            //Since you want to loop again because there could be multiple
            //messages in that chatroom which all have a unique ID, do this loop:
            let enumerator = snapshot.children
            while let rest = enumerator.nextObject() as? FIRDataSnapshot {
               //this is 1 single message here
               let values = rest.value as? NSDictionary
               for (key, value) in values{
               print("Key: \(key), value: \(value)") 
               }
               //lets say you want to check if there is a value with a key named "text":
            let textUser = values?["text"] as? String ?? "No text found" 
               //providing a default value if there is no text. You could leave it empty, than it is nil (not recommend)
               //Not only is this a lot more readable, I do not use force unwrapping so your app 
               //cannot cause an exception.

               //Get more values from the dictionary as I did with textUser
               //Initialize message than here. You will see you do not need to force unwrap anything :D
               }

            }

        })

Ответ 2

Ваш словарь выглядит следующим образом:

▿ 1 element
  ▿ 0 : 2 elements
    - key : "-KotqLUUucaRagTRt967"
    ▿ value : 3 elements
      ▿ 0 : 2 elements
        - key : sender
        - value : eGTYRSo81JefgasYLRHUFHUTnEC3
      ▿ 1 : 2 elements
        - key : text
        - value : test
      ▿ 2 : 2 elements
        - key : timestamp
        - value : 1499914135546

Но на самом деле это должно выглядеть (если вы хотите получить доступ к словарю [ "sender" ]):

▿ 2 elements
  ▿ 0 : 2 elements
    - key : sender
    - value : eGTYRSo81JefgasYLRHUFHUTnEC3
  ▿ 1 : 2 elements
    - key : text
    - value : test
  ▿ 2 : 2 elements
    - key : timestamp
    - value : 1499914135546

Первое решение:

Вам нужно добавить .child( "- KotqLUUucaRagTRt967" ) в ваш запрос.

ИЛИ

Второе решение:

Вам нужно сделать что-то вроде этого:

query.observe(.childAdded, with: { snapshot in
    for child in snapshot.children {
        guard let value = child.value as? NSDictionary else {
            return
        }

        guard let sender = value["sender"] as? String else {
            return
        }

        // You can user the sender
    }
})

ОБНОВЛЕНИЕ:

query.observe(.childAdded, with: { snapshot in
    for child in snapshot.children.allObjects as! [FIRDataSnapshot] {
        if let value = child.value as? [String:Any], let sender = value["sender"] as? String {  
            // You can user the sender
        }
    }
})

Примечание

Я заменил observSingleEvent на наблюдать, поскольку Frank van Puffelen сказал, что это необычная комбинация для .childAdded.

Ответ 3

Как объясняется Pipiks, ваша проблема заключается в том, что вы пытаетесь получить доступ к отдельным сообщениям на одном уровне выше возвращаемых данных.

Значение моментального снимка дает вам словарь сообщений чата, где клавишами верхнего уровня являются клавиши сообщения чата. Чтобы отобразить возвращаемые данные в массив сообщений, я бы использовал следующий код:

query.observeSingleEvent(of: .childAdded, with: { snapshot in

    guard let messagesDict = snapshot.value as? [String: AnyObject] else { return }

    self.messages = messagesDict.flatMap({ (messageId: String, messageData: Any) -> Message? in
        guard
            let sender = messageData["sender"] as? String,
            let text = messageData["text"] as? String,
            let timestamp = messageData["timestamp"] as? Int,
            let message = Message(key: messageKey, sender: sender, text: text, timestamp: timestamp) else {
                return nil
        }
        return message
    })

    self.tableView.reloadData()
})

Что это значит, так это отображение вашего словаря сообщений в массив объектов Message.

Я использовал flatMap для фильтрации любых сообщений, которые не имеют значений отправителя, текста или метки времени (поэтому flatMap возвращает объект [Message]).

Помогает ли это решить проблему?