NSCocoa Ошибка при получении данных JSON из PHP-скрипта с использованием Swift - PullRequest
0 голосов
/ 28 октября 2018

Я работал в течение нескольких месяцев над приложением для iPhone, которое извлекает / обновляет информацию из базы данных MYSQL, используя PHP-скрипты, которые преобразуют любые результаты в данные JSON.Пытаясь интегрировать мои PHP-скрипты с моим приложением, я сталкиваюсь с некоторыми проблемами при получении данных JSON, переданных обратно в приложение посредством PHP-скрипта.Это текст скрипта, который, как я проверял, работает.

<?php
//Read zip code from POST request
$queryzip = $_POST["ZipID"];

//Create MYSQL connection
$conn = new mysqli("localhost", "root", "Gov4Me", "Gov4Me");

//Check connection
if (mysqli_connect_error()) {
    die("Database connection failed: " .mysqli_connect_error());
}

//Query DB for FedConOfficeModel results using POST ZipID
$sql = "SELECT FedCon.FedConName, FedConOffice.FedConOfficeAddress,                                                 FedConOffice.FedConOfficeAddress2, Cities.City, Cities.ZipID, FedConOffice.Phone, ConDistricts.ConDistrict AS FedConDistrict, FedConOffice.Latitude, FedConOffice.Longitude
    FROM ConDistricts JOIN FedCon ON ConDistricts.ConDistrict = FedCon.FedConDistrict
        JOIN FedConOffice ON FedConOffice.FedConName = FedCon.FedConName
        JOIN Cities ON Cities.ZipID = FedConOffice.ZipID
    WHERE ConDistricts.ZipID = '$queryzip'";

$sqlresults = mysqli_query($conn, "SELECT FedCon.FedConName, FedConOffice.FedConOfficeAddress,                                                 FedConOffice.FedConOfficeAddress2, Cities.City, Cities.ZipID, FedConOffice.Phone, ConDistricts.ConDistrict AS FedConDistrict, FedConOffice.Latitude, FedConOffice.Longitude
    FROM ConDistricts JOIN FedCon ON ConDistricts.ConDistrict = FedCon.FedConDistrict
        JOIN FedConOffice ON FedConOffice.FedConName = FedCon.FedConName
        JOIN Cities ON Cities.ZipID = FedConOffice.ZipID
    WHERE ConDistricts.ZipID = '$queryzip'")
    or die(mysqli_error($conn));

//Check if there are results
if ($result = mysqli_query($conn, $sql)){
    //If so, then create a results array and a temporary one to hold the data
    $resultArray = array();
    $tempArray = array();

    //Loop through each row in the result set
    while($row = $result->fetch_object()){
        //Add each row into our results array
        $tempArray = $row;
        array_push($resultArray, $tempArray);
    }

    //Finally, encode the array to JSON and output the results
    echo json_encode($resultArray);
}

//Close connections
mysqli_close($conn);
?>

Когда я запускаю этот скрипт и передаю ему действительный почтовый индекс через POST, вывод выглядит так:

[{"FedConName":"Adrian Smith","FedConOfficeAddress":"1811 West Second Street","FedConOfficeAddress2":"Suite 275","City":"Grand Island","ZipID":"68803","Phone":"(308) 384-3900","FedConDistrict":"NE-3","Latitude":"40.917091","Longitude":"-98.358109"},{"FedConName":"Adrian Smith","FedConOfficeAddress":"416 Valley View Drive","FedConOfficeAddress2":"Suite 600","City":"Scottsbluff","ZipID":"69361","Phone":"(308) 633-6333","FedConDistrict":"NE-3","Latitude":"41.878609","Longitude":"-103.656822"}]

Вот файл Swift, в котором этот скрипт называется.

//
// FedConRetrieve.Swift
//
// Created by John Tree on 10/20/2018
// Copyright © 2018 John Tree. All rights reserved.
//

import UIKit
import Foundation

protocol FedConRetrieveProtocol: class{
    func itemsDownloaded(items: NSArray)
}

class FedConRetrieve: NSObject, URLSessionDelegate {
    //properties

    //Declare a delegate property
    weak var delegate: FedConRetrieveProtocol!
    //Declare data variable to store incoming bits
    var data = Data()
    //Declare urlPath to phpservice
    let urlPath: String = "http://localhost/~johntree/FedConOfficeQuery.php"

    //Define function to retrieve data from MYSQL and handle it
    func downloadItems(){
        //Define function local variable to hold value from urlPath variable
        let url: URL = URL(string: urlPath)!
        //Instantiate a URLSession object, with a default configuration
        let defaultSession = Foundation.URLSession(configuration: URLSessionConfiguration.default)
        //Assign a data task in the URLSession using the local url variable
        let task = defaultSession.dataTask(with: url){(data, response, error)
            in
                //If there is an error, notify of failure
                if error != nil{
                    print("Failed to download data")
                }
                //If there are no errors, notify success and parse JSON data
                else{
                    print("Data downloaded")
                    self.parseJSON(_data: data!)
                }
        }   
        task.resume()
    }

    func parseJSON(_data: Data){
        //Create an array variable to hold the data
        var jsonResult = NSArray()
        //Use JSON data to return a foundation object
        do{
            jsonResult = try JSONSerialization.jsonObject(with: data, options: JSONSerialization.ReadingOptions.allowFragments) as! NSArray
        }
        //If there is an error, print it to the console
        catch let error as NSError{
            print(error)
        }

        //Create an NSDictionary to hold key-value pairs
        var jsonElement = NSDictionary()
        //Create an array to hold each instance of the FedConOffice model
        let offices = NSMutableArray()

        //Loop through the JSON result Array to the last entry
        for i in 0 ..< jsonResult.count{
            //For each line in the JSON data, add it as a key-value pair to the dictionary
            jsonElement = jsonResult[i] as! NSDictionary
            //Instantiate a FedConOfficeModel object to hold the values from the DB
            let office = FedConOfficeModel()

        //Ensure none of the JsonElement values are nil, assign Dictionary entries to the variable names declared
        if let FedConName = jsonElement["FedConName"] as? String,
            let FedConOfficeAddress = jsonElement["FedConOfficeAddress"] as? String,
            let FedConOfficeAddress2 = jsonElement["FedConOfficeAddress2"] as? String,
            let City = jsonElement["City"] as? String,
            let ZipID = jsonElement["ZipID"] as? String,
            let Phone = jsonElement["Phone"] as? String,
            let Latitude = jsonElement["Latitude"] as? String,
            let Longitude = jsonElement["Longitude"] as? String
            {
                //Use the above variable names to assign values to the FedConOfficeModel object's properties
                office.FedConName = FedConName
                office.FedConOfficeAddress = FedConOfficeAddress
                office.FedConOfficeAddress2 = FedConOfficeAddress2
                office.City = City
                office.ZipID = ZipID
                office.Phone = Phone
                office.Latitude = Latitude
                office.Longitude = Longitude
            }
            //Add the fully populate FedConOfficeModel to the Array
            offices.add(office)
            print(offices.description)
        }

        //Pass the offices array to the protocol method for the view controller to use
        DispatchQueue.main.async(execute: { () -> Void in self.delegate.itemsDownloaded(items: offices)
        })
    }
}

Файл Swift, который служит моделью данных для офисов

//
//  FedConOfficeModel.swift
// Gov4Me
//
// Created by John Tree on 10/20/18.
// Copyright © 2018 John Tree. All rights reserved.
//

import UIKit

class FedConOfficeModel: NSObject {
    //properties
    var FedConName: String?
    var FedConOfficeAddress: String?
    var FedConOfficeAddress2: String?
    var City: String?
    var ZipID: String?
    var Phone: String?
    var FedConDistrict: String?
    var Latitude: String?
    var Longitude: String?

    //empty constructor
    override init()
    {

    }

    //construct with variable values
    init(FedConName: String, FedConOfficeAddress: String, FedConOfficeAddress2: String, City: String, ZipID: String, Phone: String, FedConDistrict: String, Latitude: String, Longitude: String)
    {
        self.FedConName = FedConName
        self.FedConOfficeAddress = FedConOfficeAddress
        self.FedConOfficeAddress2 = FedConOfficeAddress2
        self.City = City
        self.ZipID = ZipID
        self.Phone = Phone
        self.FedConDistrict = FedConDistrict
        self.Latitude = Latitude
        self.Longitude = Longitude  
    }

    //print objects current state
    override var description: String
    {
        return "Name: \(String(describing: FedConName))," +
               "Address: \(String(describing: FedConOfficeAddress))," +
               "Address Line 2: \(String(describing: FedConOfficeAddress2))," +
               "City: \(String(describing: City))," +
               "Zip Code: \(String(describing: ZipID))," +
               "Phone: \(String(describing: Phone))," +
               "Congressional District: \(String(describing: FedConDistrict))," +
               "Latitude: \(String(describing: Latitude))," +
               "Longitude: \(String(describing: Longitude))"
    }

}

И Swiftфайл, который получает данные от вызова.

//
//  MyElectedOfficials.swift
//  Gov4Me App
//
//  Created by John Tree on 8/25/18.
//  Copyright © 2018 John Tree. All rights reserved.
//

import UIKit
import CoreLocation
import Alamofire

class MyElectedOfficials: UIViewController, UITextFieldDelegate, CLLocationManagerDelegate
{

    //Properties
    //Variable to hold query URL selection from previous view
    var queryURL = String()

    //Outlet for textfield
    @IBOutlet weak var ZipField: UITextField!
    //Variable to hold user input
    lazy var postZipID: String = ZipField.text!
    //Variable to hold DB response in label
    @IBOutlet weak var responseLabel: UILabel!


    //Create a variable to contain the gradient layer data
    var gradientLayer : CAGradientLayer!

    //Set status bar to light text
    override var preferredStatusBarStyle: UIStatusBarStyle {
        return .lightContent
    }

    //Outlet for submit button
    @IBAction func Submit(_ sender: Any) {
        let url = URL(string: queryURL)!
            var request = URLRequest(url: url)
            request.httpMethod = "POST"
            request.setValue("application/x-www-form-urlencoded", forHTTPHeaderField: "Content-Type")
        let postString = "ZipID=" + postZipID
            print(postString)
            request.httpBody = postString.data(using: .utf8)
            let task = URLSession.shared.dataTask(with: request) { data, response, error in
                guard let data = data, error == nil else {
                    //check for fundamental networking error
                    print("error=\(String(describing: error))")
                    return
                }

                if let httpStatus = response as? HTTPURLResponse, httpStatus.statusCode != 200 {
                    //check for http errors
                    print("statusCode should be 200, but is \(httpStatus.statusCode)")
                    print("response = \(String(describing: response))")
                }

                let responseString = String(data: data, encoding: .utf8)
                print("responseString =" + responseString!)

                self.responseLabel.text = responseString
            }

            task.resume()

        performSegue(withIdentifier: "Officials1-Officials2", sender: self)
    }


   //Outlet for Back Button
    @IBAction func backbuttontap(_ sender: Any) {
        performSegue(withIdentifier: "Elected1-Pick", sender: self)
    }

    func createGradientLayer(){
        gradientLayer = CAGradientLayer()
        //Initialize gradienLayer object

        gradientLayer.frame=self.view.bounds
        //Set frame to view controller's bounds

        gradientLayer.colors = [UIColor.green.cgColor, UIColor.cyan.cgColor]
        //Create an array to hold the gradient colors

        self.view.layer.insertSublayer(gradientLayer, at: 0)
        //Add the gradient layer as a sublayer to the view
    }

    override func viewWillAppear(_ animated: Bool){
        super.viewWillAppear(animated)
        createGradientLayer()
        //Before the view is added to the hierarchy, draw the gradient background
    }

    override func viewDidLoad(){
        super.viewDidLoad()
        ZipField.delegate = self
        // Do any additional setup after loading the view.
    }

    override func didReceiveMemoryWarning(){
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }

    func textFieldShouldReturn(_ textField: UITextField) -> Bool{
        textField.resignFirstResponder()
        performSegue(withIdentifier: "Officials1-Officials2", sender: self)
        return true
    }



    /*
    // MARK: - Navigation

    // In a storyboard-based application, you will often want to do a little preparation before navigation
    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        // Get the new view controller using segue.destinationViewController.
        // Pass the selected object to the new view controller.
    }
    */

}

Когда я пытаюсь запустить это приложение, оно не падает, но данные не заполняются в моем приложении.Я знаю, что проблема лежит где-то в моем файле FedConRetrieve.Swift, но я чувствовал, что другие нужны для контекста.Журнал из консоли в xcode дает следующий вывод, когда я пытаюсь все вместе.

2018-10-28 09:00:59.535344-0400 Gov4Me App[24587:13858262] libMobileGestalt MobileGestalt.c:890: MGIsDeviceOneOfType is not supported on this platform.
2018-10-28 09:01:03.641829-0400 Gov4Me App[24587:13858262] [MC] System group container for systemgroup.com.apple.configurationprofiles path is /Users/johntree/Library/Developer/CoreSimulator/Devices/D2A0C1BC-26A5-4C06-802B-71105AC6345F/data/Containers/Shared/SystemGroup/systemgroup.com.apple.configurationprofiles
2018-10-28 09:01:03.726773-0400 Gov4Me App[24587:13858262] [MC] Reading from private effective user settings.
ZipID=68803
Data downloaded
Error Domain=NSCocoaErrorDomain Code=3840 "No value." UserInfo={NSDebugDescription=No value.}
responseString =[{"FedConName":"Adrian Smith","FedConOfficeAddress":"1811 West Second Street","FedConOfficeAddress2":"Suite 275","City":"Grand Island","ZipID":"68803","Phone":"(308) 384-3900","FedConDistrict":"NE-3","Latitude":"40.917091","Longitude":"-98.358109"},{"FedConName":"Adrian Smith","FedConOfficeAddress":"416 Valley View Drive","FedConOfficeAddress2":"Suite 600","City":"Scottsbluff","ZipID":"69361","Phone":"(308) 633-6333","FedConDistrict":"NE-3","Latitude":"41.878609","Longitude":"-103.656822"}]
=================================================================
Main Thread Checker: UI API called on a background thread: -[UILabel setText:]
PID: 24587, TID: 13858294, Thread name: (none), Queue name: NSOperationQueue 0x6000020fce20 (QOS: UNSPECIFIED), QoS: 0
Backtrace:
4   Gov4Me App                          0x0000000106a33f78 $S10Gov4Me_App18MyElectedOfficialsC6SubmityyypFy10Foundation4DataVSg_So13NSURLResponseCSgs5Error_pSgtcfU_ + 2664
5   Gov4Me App                          0x0000000106a3420d $S10Gov4Me_App18MyElectedOfficialsC6SubmityyypFy10Foundation4DataVSg_So13NSURLResponseCSgs5Error_pSgtcfU_TA + 13
6   Gov4Me App                          0x0000000106a1f820 $S10Foundation4DataVSgSo13NSURLResponseCSgs5Error_pSgIegggg_So6NSDataCSgAGSo7NSErrorCSgIeyByyy_TR + 336
7   CFNetwork                           0x0000000109784ef0 __75-[__NSURLSessionLocal taskForClass:request:uploadFile:bodyData:completion:]_block_invoke + 19
8   CFNetwork                           0x000000010979b0bc __49-[__NSCFLocalSessionTask _task_onqueue_didFinish]_block_invoke + 172
9   Foundation                          0x0000000106f8119d __NSBLOCKOPERATION_IS_CALLING_OUT_TO_A_BLOCK__ + 7
10  Foundation                          0x0000000106f810a4 -[NSBlockOperation main] + 68
11  Foundation                          0x0000000106f7de13 -[__NSOperationInternal _start:] + 689
12  Foundation                          0x0000000106f83e4a __NSOQSchedule_f + 227
13  libdispatch.dylib                   0x000000010ac1e5d1 _dispatch_call_block_and_release + 12
14  libdispatch.dylib                   0x000000010ac1f63e _dispatch_client_callout + 8
15  libdispatch.dylib                   0x000000010ac22589 _dispatch_continuation_pop + 565
16  libdispatch.dylib                   0x000000010ac21963 _dispatch_async_redirect_invoke + 859
17  libdispatch.dylib                   0x000000010ac30028 _dispatch_root_queue_drain + 351
18  libdispatch.dylib                   0x000000010ac309cd _dispatch_worker_thread2 + 130
19  libsystem_pthread.dylib             0x000000010b006169 _pthread_wqthread + 1387
20  libsystem_pthread.dylib             0x000000010b005be9 start_wqthread + 13
2018-10-28 09:01:06.956276-0400 Gov4Me App[24587:13858294] [reports] Main Thread Checker: UI API called on a background thread: -[UILabel setText:]
PID: 24587, TID: 13858294, Thread name: (none), Queue name: NSOperationQueue 0x6000020fce20 (QOS: UNSPECIFIED), QoS: 0
Backtrace:
4   Gov4Me App                          0x0000000106a33f78 $S10Gov4Me_App18MyElectedOfficialsC6SubmityyypFy10Foundation4DataVSg_So13NSURLResponseCSgs5Error_pSgtcfU_ + 2664
5   Gov4Me App                          0x0000000106a3420d $S10Gov4Me_App18MyElectedOfficialsC6SubmityyypFy10Foundation4DataVSg_So13NSURLResponseCSgs5Error_pSgtcfU_TA + 13
6   Gov4Me App                          0x0000000106a1f820 $S10Foundation4DataVSgSo13NSURLResponseCSgs5Error_pSgIegggg_So6NSDataCSgAGSo7NSErrorCSgIeyByyy_TR + 336
7   CFNetwork                           0x0000000109784ef0 __75-[__NSURLSessionLocal taskForClass:request:uploadFile:bodyData:completion:]_block_invoke + 19
8   CFNetwork                           0x000000010979b0bc __49-[__NSCFLocalSessionTask _task_onqueue_didFinish]_block_invoke + 172
9   Foundation                          0x0000000106f8119d __NSBLOCKOPERATION_IS_CALLING_OUT_TO_A_BLOCK__ + 7
10  Foundation                          0x0000000106f810a4 -[NSBlockOperation main] + 68
11  Foundation                          0x0000000106f7de13 -[__NSOperationInternal _start:] + 689
12  Foundation                          0x0000000106f83e4a __NSOQSchedule_f + 227
13  libdispatch.dylib                   0x000000010ac1e5d1 _dispatch_call_block_and_release + 12
14  libdispatch.dylib                   0x000000010ac1f63e _dispatch_client_callout + 8
15  libdispatch.dylib                   0x000000010ac22589 _dispatch_continuation_pop + 565
16  libdispatch.dylib                   0x000000010ac21963 _dispatch_async_redirect_invoke + 859
17  libdispatch.dylib                   0x000000010ac30028 _dispatch_root_queue_drain + 351
18  libdispatch.dylib                   0x000000010ac309cd _dispatch_worker_thread2 + 130
19  libsystem_pthread.dylib             0x000000010b006169 _pthread_wqthread + 1387
20  libsystem_pthread.dylib             0x000000010b005be9 start_wqthread + 13

Вы можете видеть, что строка ответа JSON содержит допустимые отформатированные данные JSON, но она никогда не добавляется в мою модель данных иЯ не смог понять, почему.В течение двух недель пытались разобраться с этой проблемой, но безрезультатно.Если кто-то может предоставить какую-либо информацию, я был бы очень признателен.

Я нашел одно решение для потоков, которое я попробовал, но оно не сработало, установив значение заголовка типа содержимого для использования JSON,

request.addValue("application/json", forHTTPHeaderField: "Content-Type")

но когда я пытаюсь использовать этот метод, строка ответа возвращается совершенно пустой.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...