将音频从 Swift 应用程序发送到 PHP 服务器,但音频丢失了
Sending audio from a Swift App to PHP Server, and somewhere the audio is lost
我正在 Swift 中制作一个应用程序来录制一些音频,然后将该录音发送到我的 PHP 服务器。
应用程序录制的音频片段很好(可以毫无问题地播放)。当我 println
录制的音频剪辑时,它会显示负载和字节数据负载(当我将音频放入 NSData
包装器时也是如此)。这一切都向我表明应用程序内的音频很好。
在我的服务器上捕捉记录的 PHP 文件也工作正常,没有错误。
但是录制的音频剪辑丢失了。
Swift上传录音的代码:
// The variable "recordedFileURL" is defined earlier in the code like this:
currentFilename = "xxxx.m4a"
let dirPaths = NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true)
let docsDir: AnyObject=dirPaths[0]
recordedFilePath = docsDir.stringByAppendingPathComponent(self.currentFilename)
recordedFileURL = NSURL(fileURLWithPath: self.recordedFilePath)
// "currentFilename", "recordedFilePath" and "recordedFileURL" are all global variables
// This recording stored at "recordedFileURL" can be played back fine.
let sendToPath = "http://......../catch.php"
let sendToURL = NSURL(string: sendToPath)
let recording: NSData? = NSData(contentsOfURL: recordedFileURL)
let boundary = "--------14737809831466499882746641449----"
let contentType = "multipart/form-data;boundary=\(boundary)"
var request = NSMutableURLRequest()
request.URL = sendToURL
request.HTTPMethod = "POST"
request.addValue(contentType, forHTTPHeaderField: "Content-Type")
request.addValue(recId, forHTTPHeaderField: "REC-ID") // recId is defined elsewhere
var body = NSMutableData()
var header = "Content-Disposition: form-data; name=\"\(currentFilename)\"; filename=\"\(recordedFilePath)\"\r\n"
body.appendData(("\r\n-\(boundary)\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData((header as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData(("Content-Type: application/octet-stream\r\n\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData(recording!) // adding the recording here
body.appendData(("\r\n-\(boundary)\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
request.HTTPBody = body
var session = NSURLSession.sharedSession()
var task = session.dataTaskWithRequest(request, completionHandler: { (data, response, error) -> Void in
println("upload complete")
let dataStr = NSString(data: data, encoding: NSUTF8StringEncoding)
println(dataStr)
})
task.resume()
PHP 文件 catch.php
中应该接收录音的代码:
$contents = file_get_contents('php://input');
$files = $_FILES;
echo "Caught the following:/r/n";
echo "Contents:" . var_export($contents) . "/r/n";
echo "Files:" . var_export($files) . "/r/n";
每当我 运行 所有这些时,我都会从 catch.php
得到以下输出:
Caught the following:
Contents:''
Files:array (
)
所以 catch.php
根本没有收到任何东西。
是我发错了录音,还是接错了录音?或者两者兼而有之?
提前致谢。
您的 PHP 代码基本没问题。 $_FILES
部分没问题,但是 php://input
is not available with enctype="multipart/form-data"
.
问题在于您如何在 Swift 代码中生成 HTTP 请求。主要是 HTTP headers。
当为多部分数据创建 headers 时,模式是这样的(如果我们选择 AAAAA 作为我们的边界):
- 我们选择的边界:"AAAAA"
- 内容类型 = "multipart/form-data;boundary=AAAAA"
- 起始边界 = --AAAAA
- 结束边界 = --AAAAA--
所以稍微修改一下代码:
// This was your main problem
let boundary = "--------14737809831466499882746641449----"
let beginningBoundary = "--\(boundary)"
let endingBoundary = "--\(boundary)--"
let contentType = "multipart/form-data;boundary=\(boundary)"
// recordedFilePath is Optional, so the resulting string will end up being 'Optional("/path/to/file/filename.m4a")', which is wrong.
// We could just use currentFilename if we wanted
let filename = recordedFilePath ?? currentFilename
var header = "Content-Disposition: form-data; name=\"\(currentFilename)\"; filename=\"\(recordedFilePath!)\"\r\n"
var body = NSMutableData()
body.appendData(("\(beginningBoundary)\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData((header as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData(("Content-Type: application/octet-stream\r\n\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData(recording!) // adding the recording here
body.appendData(("\r\n\(endingBoundary)\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
var request = NSMutableURLRequest()
request.URL = sendToURL
request.HTTPMethod = "POST"
request.addValue(contentType, forHTTPHeaderField: "Content-Type")
request.addValue(recId, forHTTPHeaderField: "REC-ID") // recId is defined elsewhere
request.HTTPBody = body
万一你 运行 再次遇到这样的事情,在调试这样的网络代码时,我喜欢使用让我检查正在传输的 HTTP 网络数据的工具运行。我个人喜欢HTTPScoop because its simple, but you can also use BurpSuite or Charles
我只是 运行 你的代码,并将 HTTP 流量与我使用 curl 发出请求时发生的情况进行了比较
curl -X POST http://localhost/\~cjwirth/catch.php -F "file=@Untitled.m4a"
这里是swift4~5代码。
在 ViewController 和 link 按钮操作上为 @IBAction 和 @IBOutlet 创建新的按钮(代码中的 buttonLabel)
[按住重新编码并释放上传音频文件]
import UIKit
import AVFoundation
class ViewController2: UIViewController, AVAudioRecorderDelegate{
var recordingSession: AVAudioSession!
var audioRecorder: AVAudioRecorder!
var audioPlayer: AVAudioPlayer!
var numberOfRecords = 0
@IBOutlet weak var buttonLabel: UIButton!
let E_401 = "E_401"
let DATABASE_PATH = "http://<IP_address_of_PHP_server>/YourPrjectName/"
override func viewDidLoad() {
super.viewDidLoad()
// Set the recognizer to recognize the button action
let longPressRecognizer = UILongPressGestureRecognizer(target: self, action: #selector(record))
longPressRecognizer.minimumPressDuration = 0
buttonLabel.addGestureRecognizer(longPressRecognizer)
// Setting up session
recordingSession = AVAudioSession.sharedInstance()
// Get permission from user to use mic
AVAudioSession.sharedInstance().requestRecordPermission{ (hasPermission) in
if hasPermission
{print ("ACCEPTED")}
}
}
@IBAction func record(_ gestureRecognizer: UILongPressGestureRecognizer) {
// Check if we have an active recorder
if (gestureRecognizer.state == .began) && (audioRecorder == nil) {
// Increase +1 total number of recordings for every new recording made
self.numberOfRecords += 1
// Setting filename and settings
let filename = getDirectory().appendingPathComponent("\(numberOfRecords).m4a")
let settings = [
AVFormatIDKey: Int(kAudioFormatMPEG4AAC),
AVSampleRateKey: 12000,
AVNumberOfChannelsKey: 1,
AVEncoderAudioQualityKey: AVAudioQuality.medium.rawValue
]
do
{
// Start audio recording
buttonLabel.setTitle("Recording...", for: .normal)
audioRecorder = try AVAudioRecorder(url: filename, settings: settings)
audioRecorder.delegate = self
audioRecorder.record()
}
catch
{
// Catch for errors
displayAlert(title: "Oops!", message: "Recording failed")
}
} else if gestureRecognizer.state == .ended && (audioRecorder != nil)
{
// Stopping audio recording
buttonLabel.setTitle("Start Recording", for: .normal)
audioRecorder.stop()
audioRecorder = nil
do {
let filename = getDirectory().appendingPathComponent("\(numberOfRecords).m4a")
let recording: NSData = try NSData(contentsOf: filename)
self.uploadFile(fileData: recording as Data, fileName: "\(numberOfRecords).m4a"){
(fileURL, e) in
if e == nil {
print("FILE URL: " + fileURL!)
}
}
} catch {
print("Unexpected <<<<<<<<<<<<<<>>>>>>>>>>>>>> error: \(error)")
}
}
}
// Function that gets path to directory
func getDirectory () -> URL
{
let paths = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)
let documentDirectory = paths[0]
return documentDirectory
}
// Function that displays an alert
func displayAlert(title:String, message:String)
{
let alert = UIAlertController(title: title, message: message, preferredStyle: .alert)
alert.addAction(UIAlertAction(title: "dismiss", style: .default, handler: nil))
present(alert, animated: true, completion: nil)
}
func uploadFile(fileData:Data, fileName:String , completion: @escaping (_ fileURL:String?, _ error:String?) -> Void) {
let recId = "\(numberOfRecords)"
print("FILENAME: \(fileName)")
let request = NSMutableURLRequest()
let boundary = "--------14737809831466499882746641449----"
let beginningBoundary = "--\(boundary)"
let endingBoundary = "--\(boundary)--"
let contentType = "multipart/form-data;boundary=\(boundary)"
request.url = URL(string: DATABASE_PATH + "catch.php")
// catch.php is php script on server
request.httpShouldHandleCookies = false
request.timeoutInterval = 60
request.httpMethod = "POST"
request.setValue(contentType, forHTTPHeaderField: "Content-Type")
let body = NSMutableData()
body.append("--\(boundary)\r\n".data(using: String.Encoding.utf8)!)
body.append("Content-Disposition: form-data; name=\"fileName\"\r\n\r\n".data(using: String.Encoding.utf8)!)
body.append("\(fileName)\r\n".data(using: String.Encoding.utf8)!)
body.append("--\(boundary)\r\n".data(using: String.Encoding.utf8)!)
body.append("Content-Disposition: form-data; name=\"file\"; filename=\"file\"\r\n".data(using: String.Encoding.utf8)!)
body.append(("\(beginningBoundary)\r\n" as NSString).data(using: String.Encoding.utf8.rawValue)!)
body.append(("Content-Type: application/octet-stream\r\n\r\n" as NSString).data(using: String.Encoding.utf8.rawValue)!)
body.append(fileData)
body.append("\r\n".data(using: String.Encoding.utf8)!)
body.append("--\(boundary)--\r\n".data(using: String.Encoding.utf8)!)
request.addValue(contentType, forHTTPHeaderField: "Content-Type")
// request.addValue(recId, forHTTPHeaderField: "REC-ID")
request.httpBody = body as Data
let session = URLSession.shared
let task = session.dataTask(with: request as URLRequest) { (data, response, error) in
guard let _:Data = data as Data?, let _:URLResponse = response, error == nil else {
DispatchQueue.main.async { completion(nil, error!.localizedDescription) }
return
}
if let response = String(data: data!, encoding: String.Encoding(rawValue: String.Encoding.utf8.rawValue)) {
print("XSUploadFile -> RESPONSE: " + self.DATABASE_PATH + response)
DispatchQueue.main.async { completion(self.DATABASE_PATH + response, nil) }
// NO response
} else { DispatchQueue.main.async { completion(nil, self.E_401) } }// ./ If response
}; task.resume()
}
}
我正在 Swift 中制作一个应用程序来录制一些音频,然后将该录音发送到我的 PHP 服务器。
应用程序录制的音频片段很好(可以毫无问题地播放)。当我 println
录制的音频剪辑时,它会显示负载和字节数据负载(当我将音频放入 NSData
包装器时也是如此)。这一切都向我表明应用程序内的音频很好。
在我的服务器上捕捉记录的 PHP 文件也工作正常,没有错误。
但是录制的音频剪辑丢失了。
Swift上传录音的代码:
// The variable "recordedFileURL" is defined earlier in the code like this:
currentFilename = "xxxx.m4a"
let dirPaths = NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true)
let docsDir: AnyObject=dirPaths[0]
recordedFilePath = docsDir.stringByAppendingPathComponent(self.currentFilename)
recordedFileURL = NSURL(fileURLWithPath: self.recordedFilePath)
// "currentFilename", "recordedFilePath" and "recordedFileURL" are all global variables
// This recording stored at "recordedFileURL" can be played back fine.
let sendToPath = "http://......../catch.php"
let sendToURL = NSURL(string: sendToPath)
let recording: NSData? = NSData(contentsOfURL: recordedFileURL)
let boundary = "--------14737809831466499882746641449----"
let contentType = "multipart/form-data;boundary=\(boundary)"
var request = NSMutableURLRequest()
request.URL = sendToURL
request.HTTPMethod = "POST"
request.addValue(contentType, forHTTPHeaderField: "Content-Type")
request.addValue(recId, forHTTPHeaderField: "REC-ID") // recId is defined elsewhere
var body = NSMutableData()
var header = "Content-Disposition: form-data; name=\"\(currentFilename)\"; filename=\"\(recordedFilePath)\"\r\n"
body.appendData(("\r\n-\(boundary)\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData((header as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData(("Content-Type: application/octet-stream\r\n\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData(recording!) // adding the recording here
body.appendData(("\r\n-\(boundary)\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
request.HTTPBody = body
var session = NSURLSession.sharedSession()
var task = session.dataTaskWithRequest(request, completionHandler: { (data, response, error) -> Void in
println("upload complete")
let dataStr = NSString(data: data, encoding: NSUTF8StringEncoding)
println(dataStr)
})
task.resume()
PHP 文件 catch.php
中应该接收录音的代码:
$contents = file_get_contents('php://input');
$files = $_FILES;
echo "Caught the following:/r/n";
echo "Contents:" . var_export($contents) . "/r/n";
echo "Files:" . var_export($files) . "/r/n";
每当我 运行 所有这些时,我都会从 catch.php
得到以下输出:
Caught the following:
Contents:''
Files:array (
)
所以 catch.php
根本没有收到任何东西。
是我发错了录音,还是接错了录音?或者两者兼而有之?
提前致谢。
您的 PHP 代码基本没问题。 $_FILES
部分没问题,但是 php://input
is not available with enctype="multipart/form-data"
.
问题在于您如何在 Swift 代码中生成 HTTP 请求。主要是 HTTP headers。 当为多部分数据创建 headers 时,模式是这样的(如果我们选择 AAAAA 作为我们的边界):
- 我们选择的边界:"AAAAA"
- 内容类型 = "multipart/form-data;boundary=AAAAA"
- 起始边界 = --AAAAA
- 结束边界 = --AAAAA--
所以稍微修改一下代码:
// This was your main problem
let boundary = "--------14737809831466499882746641449----"
let beginningBoundary = "--\(boundary)"
let endingBoundary = "--\(boundary)--"
let contentType = "multipart/form-data;boundary=\(boundary)"
// recordedFilePath is Optional, so the resulting string will end up being 'Optional("/path/to/file/filename.m4a")', which is wrong.
// We could just use currentFilename if we wanted
let filename = recordedFilePath ?? currentFilename
var header = "Content-Disposition: form-data; name=\"\(currentFilename)\"; filename=\"\(recordedFilePath!)\"\r\n"
var body = NSMutableData()
body.appendData(("\(beginningBoundary)\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData((header as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData(("Content-Type: application/octet-stream\r\n\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
body.appendData(recording!) // adding the recording here
body.appendData(("\r\n\(endingBoundary)\r\n" as NSString).dataUsingEncoding(NSUTF8StringEncoding)!)
var request = NSMutableURLRequest()
request.URL = sendToURL
request.HTTPMethod = "POST"
request.addValue(contentType, forHTTPHeaderField: "Content-Type")
request.addValue(recId, forHTTPHeaderField: "REC-ID") // recId is defined elsewhere
request.HTTPBody = body
万一你 运行 再次遇到这样的事情,在调试这样的网络代码时,我喜欢使用让我检查正在传输的 HTTP 网络数据的工具运行。我个人喜欢HTTPScoop because its simple, but you can also use BurpSuite or Charles
我只是 运行 你的代码,并将 HTTP 流量与我使用 curl 发出请求时发生的情况进行了比较
curl -X POST http://localhost/\~cjwirth/catch.php -F "file=@Untitled.m4a"
这里是swift4~5代码。 在 ViewController 和 link 按钮操作上为 @IBAction 和 @IBOutlet 创建新的按钮(代码中的 buttonLabel) [按住重新编码并释放上传音频文件]
import UIKit
import AVFoundation
class ViewController2: UIViewController, AVAudioRecorderDelegate{
var recordingSession: AVAudioSession!
var audioRecorder: AVAudioRecorder!
var audioPlayer: AVAudioPlayer!
var numberOfRecords = 0
@IBOutlet weak var buttonLabel: UIButton!
let E_401 = "E_401"
let DATABASE_PATH = "http://<IP_address_of_PHP_server>/YourPrjectName/"
override func viewDidLoad() {
super.viewDidLoad()
// Set the recognizer to recognize the button action
let longPressRecognizer = UILongPressGestureRecognizer(target: self, action: #selector(record))
longPressRecognizer.minimumPressDuration = 0
buttonLabel.addGestureRecognizer(longPressRecognizer)
// Setting up session
recordingSession = AVAudioSession.sharedInstance()
// Get permission from user to use mic
AVAudioSession.sharedInstance().requestRecordPermission{ (hasPermission) in
if hasPermission
{print ("ACCEPTED")}
}
}
@IBAction func record(_ gestureRecognizer: UILongPressGestureRecognizer) {
// Check if we have an active recorder
if (gestureRecognizer.state == .began) && (audioRecorder == nil) {
// Increase +1 total number of recordings for every new recording made
self.numberOfRecords += 1
// Setting filename and settings
let filename = getDirectory().appendingPathComponent("\(numberOfRecords).m4a")
let settings = [
AVFormatIDKey: Int(kAudioFormatMPEG4AAC),
AVSampleRateKey: 12000,
AVNumberOfChannelsKey: 1,
AVEncoderAudioQualityKey: AVAudioQuality.medium.rawValue
]
do
{
// Start audio recording
buttonLabel.setTitle("Recording...", for: .normal)
audioRecorder = try AVAudioRecorder(url: filename, settings: settings)
audioRecorder.delegate = self
audioRecorder.record()
}
catch
{
// Catch for errors
displayAlert(title: "Oops!", message: "Recording failed")
}
} else if gestureRecognizer.state == .ended && (audioRecorder != nil)
{
// Stopping audio recording
buttonLabel.setTitle("Start Recording", for: .normal)
audioRecorder.stop()
audioRecorder = nil
do {
let filename = getDirectory().appendingPathComponent("\(numberOfRecords).m4a")
let recording: NSData = try NSData(contentsOf: filename)
self.uploadFile(fileData: recording as Data, fileName: "\(numberOfRecords).m4a"){
(fileURL, e) in
if e == nil {
print("FILE URL: " + fileURL!)
}
}
} catch {
print("Unexpected <<<<<<<<<<<<<<>>>>>>>>>>>>>> error: \(error)")
}
}
}
// Function that gets path to directory
func getDirectory () -> URL
{
let paths = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)
let documentDirectory = paths[0]
return documentDirectory
}
// Function that displays an alert
func displayAlert(title:String, message:String)
{
let alert = UIAlertController(title: title, message: message, preferredStyle: .alert)
alert.addAction(UIAlertAction(title: "dismiss", style: .default, handler: nil))
present(alert, animated: true, completion: nil)
}
func uploadFile(fileData:Data, fileName:String , completion: @escaping (_ fileURL:String?, _ error:String?) -> Void) {
let recId = "\(numberOfRecords)"
print("FILENAME: \(fileName)")
let request = NSMutableURLRequest()
let boundary = "--------14737809831466499882746641449----"
let beginningBoundary = "--\(boundary)"
let endingBoundary = "--\(boundary)--"
let contentType = "multipart/form-data;boundary=\(boundary)"
request.url = URL(string: DATABASE_PATH + "catch.php")
// catch.php is php script on server
request.httpShouldHandleCookies = false
request.timeoutInterval = 60
request.httpMethod = "POST"
request.setValue(contentType, forHTTPHeaderField: "Content-Type")
let body = NSMutableData()
body.append("--\(boundary)\r\n".data(using: String.Encoding.utf8)!)
body.append("Content-Disposition: form-data; name=\"fileName\"\r\n\r\n".data(using: String.Encoding.utf8)!)
body.append("\(fileName)\r\n".data(using: String.Encoding.utf8)!)
body.append("--\(boundary)\r\n".data(using: String.Encoding.utf8)!)
body.append("Content-Disposition: form-data; name=\"file\"; filename=\"file\"\r\n".data(using: String.Encoding.utf8)!)
body.append(("\(beginningBoundary)\r\n" as NSString).data(using: String.Encoding.utf8.rawValue)!)
body.append(("Content-Type: application/octet-stream\r\n\r\n" as NSString).data(using: String.Encoding.utf8.rawValue)!)
body.append(fileData)
body.append("\r\n".data(using: String.Encoding.utf8)!)
body.append("--\(boundary)--\r\n".data(using: String.Encoding.utf8)!)
request.addValue(contentType, forHTTPHeaderField: "Content-Type")
// request.addValue(recId, forHTTPHeaderField: "REC-ID")
request.httpBody = body as Data
let session = URLSession.shared
let task = session.dataTask(with: request as URLRequest) { (data, response, error) in
guard let _:Data = data as Data?, let _:URLResponse = response, error == nil else {
DispatchQueue.main.async { completion(nil, error!.localizedDescription) }
return
}
if let response = String(data: data!, encoding: String.Encoding(rawValue: String.Encoding.utf8.rawValue)) {
print("XSUploadFile -> RESPONSE: " + self.DATABASE_PATH + response)
DispatchQueue.main.async { completion(self.DATABASE_PATH + response, nil) }
// NO response
} else { DispatchQueue.main.async { completion(nil, self.E_401) } }// ./ If response
}; task.resume()
}
}