I am seeing a video stream and making .mp4
files so I am doing most of this correctly. My problem is that my video files are 0kb, empty. I'm using an iOS device to control a separate device with a camera. This camera is sending a video stream to the iOS device and that stream is decoded into a CMSampleBuffer
then turned into a CVPixelBuffer
and displayed in an UIImageView
. I'm displaying the video just fine(and a separate issue is that I'm getting -12909 errors if you know anything about fixing that pls leave a comment).
I tried recording the CMSampleBuffer
objects but I was told by the compiler errors that I needed to exclude output settings. So I removed those and it saves empty files now.
When the stream starts I call this:
func beginRecording() {
handlePhotoLibraryAuth()
createFilePath()
guard let videoOutputURL = outputURL,
let vidWriter = try? AVAssetWriter(outputURL: videoOutputURL, fileType: AVFileType.mov) else {
fatalError("AVAssetWriter error")
}
let vidInput = AVAssetWriterInput(mediaType: AVMediaType.video, outputSettings: nil)
guard vidWriter.canAdd(vidInput) else {
print("Error: Cant add video writer input")
return
}
vidInput.expectsMediaDataInRealTime = true
vidWriter.add(vidInput)
guard vidWriter.startWriting() else {
print("Error: Cant write with vid writer")
return
}
vidWriter.startSession(atSourceTime: CMTime.zero)
self.videoWriter = vidWriter
self.videoWriterInput = vidInput
self.isRecording = true
print("Recording: \(self.isRecording)")
}
And this ends it:
func endRecording() {
guard let vidInput = videoWriterInput, let vidWriter = videoWriter else {
print("Error, no video writer or video input")
return
}
vidInput.markAsFinished()
vidWriter.finishWriting {
print("Finished Recording")
self.isRecording = false
guard vidWriter.status == .completed else {
print("Warning: The Video Writer status is not completed, status: \(vidWriter.status)")
return
}
print("VideoWriter status is completed")
self.saveRecordingToPhotoLibrary()
}
}
I determined my append operation on AVAssetWriterInput
is failing
Here is my current append code, I did try CMSampleBuffer first on realtime, which im not sure why didnt work. I suspect that the realtime feature only applies to the AV components of iOS devices and not other connected devices. Then I tried this which should probably work but is not. I tried both 30 and 60fps, it's supposed to be 30 though. Am I misusing CMTime? Because I was attempting to just not use CMTime and that did not work as I mentioned.
if self.videoDecoder.isRecording,
let videoPixelBuffer = self.videoDecoder.videoWriterInputPixelBufferAdaptor,
videoPixelBuffer.assetWriterInput.isReadyForMoreMediaData {
print(videoPixelBuffer.append(frame, withPresentationTime: CMTimeMake(value: self.videoDecoder.videoFrameCounter, timescale: 30)))
self.videoDecoder.videoFrameCounter += 1
}
Here was my final code solution - the final issue I had was that CMTime is used very strangely by some example projects i found on Github/Google. Also I was unable to figure out a way to send my mp4 files to the photo library - they would always be grey videos with the correct size/length. So I had to access them from the app files directory on the device.