I am very new using Swift and SwiftUI and I'm trying to write and iOS app that reads a folder, calculcate hash for each file, and then copy them to an external drive.
For now I'm trying to import one file and calculate its hash. However, I always get the same error, saying I don't have permission to view it.
Here is my code:
//
// ContentView.swift
// FileExporter
//
// Created by adrien le falher on 27/09/2020.
//
import SwiftUI
import CryptoKit
struct ContentView: View {
@State private var document: MessageDocument = MessageDocument(message: "Hello, World!")
@State private var isImporting: Bool = false
@State private var isExporting: Bool = false
@State private var isMoving: Bool = false
var body: some View {
VStack {
GroupBox(label: Text("Message:")) {
TextEditor(text: $document.message)
}
GroupBox {
HStack {
Spacer()
Button(action: { isImporting = true }, label: {
Text("Import")
})
Spacer()
Button(action: { isExporting = true }, label: {
Text("Export")
})
Spacer()
Button(action: { isMoving = true }, label: {
Text("Export")
})
Spacer()
}
}
}
.padding()
.fileExporter(
isPresented: $isExporting,
document: document,
contentType: .plainText,
defaultFilename: "Message"
) { result in
if case .success = result {
// Handle success.
} else {
// Handle failure.
}
}
.fileImporter(
isPresented: $isImporting,
allowedContentTypes: [.image],
allowsMultipleSelection: false
) { result in
do {
print("ok")
guard let selectedFile: URL = try result.get().first else { return }
guard var fileBytes : String = try hashFile(selectedFile) else { return }
//guard let message = String(data: try Data(contentsOf: selectedFile), encoding: .utf8) else { return }
let message = fileBytes
print(message)
document.message = message
} catch let error{
print(error.localizedDescription)
document.message = error.localizedDescription
}
}
}
}
extension Data {
init(reading input: InputStream) throws {
self.init()
input.open()
defer {
input.close()
}
let bufferSize = 1024
let buffer = UnsafeMutablePointer<UInt8>.allocate(capacity: bufferSize)
defer {
buffer.deallocate()
}
while input.hasBytesAvailable {
let read = input.read(buffer, maxLength: bufferSize)
if read < 0 {
//Stream error occured
throw input.streamError!
} else if read == 0 {
//EOF
break
}
self.append(buffer, count: read)
}
}
}
func hashFile (_ fileURL : URL) -> String {
print(fileURL)
var hashed = ""
do {
var fileBytes = try Data (contentsOf: fileURL)
//OU reading : InputStream(url: fileURL)!
print("Filebytes is \(fileBytes)")
print("Bytes" + String(fileBytes.base64EncodedString()))
let hashed = SHA256.hash(data: fileBytes)
return String(hashed.description)
} catch let error {
hashed = error.localizedDescription
}
return hashed
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
/*@START_MENU_TOKEN@*/Text("Hello, World!")/*@END_MENU_TOKEN@*/
}
}
It seems that .fileImporter() is new and I am having trouble finding info on it. I don't know if I should (or how) ask permissions to iOS to access files ; the file picker works so I'm not sure what I did wrong.
Any help would be appreciated.
Thank you.
You have to call the method startAccessingSecurityScopedResource()
on the URL first, before reading the content of the file. Note: don't forget to call stopAccessingSecurityScopedResource()
after you're finished!
You can find more information in the Apple documentation.
Edit:
Here is the code that worked for me:
.fileImporter(
isPresented: $isImporting,
allowedContentTypes: [.plainText],
allowsMultipleSelection: false
) { result in
do {
guard let selectedFile: URL = try result.get().first else { return }
if selectedFile.startAccessingSecurityScopedResource() {
guard let fileContent = String(data: try Data(contentsOf: selectedFile), encoding: .utf8) else { return }
defer { selectedFile.stopAccessingSecurityScopedResource() }
} else {
// Handle denied access
}
} catch {
// Handle failure.
print("Unable to read file contents")
print(error.localizedDescription)
}
}