bibscy
bibscy

Reputation: 2708

How to write an array of dictionaries to a .txt file in Xcode Project

I am retrieving the data from Firebase, then I convert the array received of type FIRDataSnapshot into an array of dictionaries of type [String:Any]. How can I output this array of dictionaries to a .txt file in my Xcode-project in an ordered list?

The output printed in the txt file should be in this format:

Booking 1:
EmailAddress: [email protected]
PhoneNumber: 94949392
.
.
Booking 4:
EmailAddress: [email protected]
PhoneNumber: 202583963

At the bottom of my question you will find the raw data how it looks in finalDictionary

I think the the logic should be like this: 1. loop through the elements of finalDictionary 2. take each element of the array, assign a count to it + break line

let finalDictionary = [String:Any]()

func startObservingDB() {

  dbRef.child(FullData.uid!).observe(.value, with: { (snapshot: FIRDataSnapshot) in

        // an instance of FireBaseData holding all bookings under currentUid
        var newBookingInfo = [FireBaseData]()

//iterate over all children under /FullData.uid! path
   for customer in snapshot.children {

      //the customer node starting with cus...
        let customerObject = customer as! FIRDataSnapshot

       //customer key
          self.customerKey = customerObject.key
             print("this is the Stripe customer that can be charged \(customerObject.key)")

   //now iterate over each booking which is located under customerObject in Firebase
     for booking in customerObject.children {


         // after each iteration through snapshot.children, create an instance of FireBaseData with  'booking' for the current iteration & assign it to bookingItem
            var bookingItem = FireBaseData(snapshot: booking as! FIRDataSnapshot)

        //assign key of the parent to each booking
                bookingItem.Key = self.customerKey

            // append the bookingItem after each iteration to newBookingInfo array
            newBookingInfo.append(bookingItem)

         } // end of  for booking in myCustomer
    } // end of  for customer in snapshot.children

         //assign newBookingInfo to global variable bookingInfo so it can be used globally within the class
         self.bookingInfo = newBookingInfo

  // sort the array in place so that the most recent date will appear first
 self.bookingInfo.sort(by: {(DateAndTimeObject_1,DateAndTimeObject_2) -> Bool in

     DateAndTimeObject_1.TimeStampDateAndTime > DateAndTimeObject_2.TimeStampDateAndTime
  })


  // convert bookingInfo of type FIrebaseData to Array of Dictionaries
   let arrayOfDictionary = self.bookingInfo.flatMap { $0.toAnyObject() as? [String:Any] }

  finalDictionary =  arrayOfDictionary
   print("arrayOfDictionary is \(arrayOfDictionary)")

 //I can write a string to a file in Xcode project using the code snippet 
 //below, but this does not solve my problem, namely to output all elements 
 //of the array as specified above with a break space after each element 
//so that they can be in a human readable format. 

 let fileName = "myBookings"
  let dir = try? FileManager.default.url(for: .documentDirectory, in: .userDomainMask, appropriateFor: nil, create: true)

  // If the directory was found, we write a file to it
     if let fileURL = dir?.appendingPathComponent(fileName).appendingPathExtension("txt") {

     // Write to the file
     let outString = "Write this text to the file"
     do {
         try outString.write(to: fileURL, atomically: true, encoding: .utf8)
     } catch {
         print("Failed writing to URL: \(fileURL), Error: " + error.localizedDescription)
    }   
}




          self.tableView.reloadData()
      }, withCancel: { (Error:Any) in
        print("Error firebase \(Error)")
    })
  } // end of startObservingDB()



func exportBookings(){

 let path = URL(string: "file:///Users/bogdanbarbulescu/Desktop/Test.txt")

        //writing
        do {  
            try (finalDictionary).description.write(to: path!, atomically: false, encoding: .utf8)
        }
        catch let error { print("The error is \(error.localizedDescription)") 
   }
}

The output of print("arrayOfDictionary is (arrayOfDictionary)") is :

arrayof Dict is [["EmailAddress": [email protected], "PhoneNumber": 94949392, "PaymentID": ch_1AVK6KLCZ34Ur7XGsbROjWhe, "BookingAmount": 37, "BookingCompleted": 0, "NoteInstructions": Take the dog out please, "insideCabinets": true, "BookingNumber": 883924391, "BookingStatusAdmin": 0, "insideFridge": true, "FrequencyName": Every 2 weeks, "EntryInstructions": keys under carpet, "FullName": James, "SuppliesAmount": 0, "FirebaseUserID": 7b1eRsa9QWhdtA5EwyD2FiPZHSh2, "TimeStampDateAndTime": 1497790800, "PostCode": W3 7RZ, "SelectedBathRow": 2, "FrequecyAmount": 22, "laundryWash": false, "StreetAddress": 11 High St, "CostToRescheduleClient": 0, "BookingStatusClient": 1, "TimeStampBookingSavedInDB": 1497618179, "FlatNumber": Conely, "SelectedBedRow": 1, "interiorWindows": 0, "CostToCancelClient": 0, "DoormanOption": Hidden-Key, "DateAndTime": Sun, 18 Jun 2017 14:00, "insideOven": true, "SuppliesName": I have cleaning supplies],

[“EmailAddress": [email protected], "PhoneNumber": 07476953923, "PaymentID": ch_1ATTRHLCZ34Ur7XGL2CGN1I3, "BookingAmount": 63, "BookingCompleted": 0, "NoteInstructions": No, "insideCabinets": true, "BookingNumber": 173009560, "BookingStatusAdmin": 1, "insideFridge": true, "FrequencyName": Every 2 weeks, "EntryInstructions": simbasdada, "CostToRescheduleAdmin": 0, "FullName": John Luch, "SuppliesAmount": 5, "FirebaseUserID": 7b1eRsa9QWhdtA5EwyD2FiPZHSh2, "TimeStampDateAndTime": 1497632400, "PostCode": IG11 6PP, "SelectedBathRow": 2, "FrequecyAmount": 33, "laundryWash": true, "StreetAddress": High St, "CostToRescheduleClient": 0, "BookingStatusClient": 1, "TimeStampBookingSavedInDB": 1497177419, "FlatNumber": Flat 11, "SelectedBedRow": 2, "interiorWindows": 0, "DoormanOption": Hidden-Key, "DateAndTime": Fri, 16 Jun 2017 18:00, "insideOven": false, "SuppliesName": Bring cleaning supplies]]

Upvotes: 0

Views: 695

Answers (1)

DonMag
DonMag

Reputation: 77568

If all you want to do is output your Array of Dictionaries in an "orderly / readable" format, you can use this:

if let orderedKeys = finalDictionary.first?.keys {

    var outputString = ""
    var i = 1

    for d in finalDictionary {
        outputString += "Booking \(i):\n"
        for k in orderedKeys {
            outputString += k + ": " + (d[k] ?? "(no value)") + "\n"
        }
        outputString += "\n"
        i += 1
    }

    print(outputString)
    // or, write outputString to a text file

}

Note: this is just quick code... not necessarily optimal or 100% error free, and assumes all records have the same keys. More of a "here's a direction to go" kinda thingy :)

Edit: If the dictionaries may have variable sets of keys, you can do this:

var outputString = ""
var i = 1
for d in finalDictionary {
    outputString += "Booking \(i):\n"
    for k in d.keys {
        outputString += k + ": " + (d[k] ?? "(no value)") + "\n"
    }
    outputString += "\n"
    i += 1
}

print(outputString)

Edit 2: probably a little "Swiftier" way of doing it:

outputString = ""
i = 1

finalDictionary.forEach {
    outputString += "Booking \(i):\n"
    $0.forEach { outputString += "\($0): \($1)\n" }
    outputString += "\n"
    i += 1
}

print(outputString)

Because Dictionaries are unordered, you may end up with:

Booking 1:
EmailAddress: [email protected]
PhoneNumber: 94949392
FullName: John
.
Booking 2:
SomeOtherKey: Some Value
FullName: Dave
EmailAddress: [email protected]
.
Booking 3:
FullName: Chris
PhoneNumber: 202583963
SomeOtherKey: Some Other Value
EmailAddress: [email protected]

and so on.

Upvotes: 1

Related Questions