iosswiftswiftuixcode12file-import

Getting The file 'xxx" couldn't be opened because you don't have permission to view it when importing


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.


Solution

  • 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)
            }
        }