Получение Невозможно открыть файл "xxx", потому что у вас нет разрешения на его просмотр при импорте.

Я очень новичок в использовании Swift и SwiftUI, и я пытаюсь написать приложение для iOS, которое читает папку, вычисляет хэш для каждого файла, а затем копирует их на внешний диск.

А пока я пытаюсь импортировать один файл и вычислить его хеш. Однако я всегда получаю одну и ту же ошибку, говоря, что у меня нет разрешения на ее просмотр.

Вот мой код:

    //
//  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@*/
    }
}

Кажется, что .fileImporter () новый, и у меня проблемы с поиском информации о нем. Я не знаю, следует ли (и как) запрашивать разрешения у iOS для доступа к файлам; средство выбора файлов работает, поэтому я не уверен, что сделал не так.

Любая помощь будет оценена по достоинству.

Спасибо.


person Sydney Writer    schedule 29.09.2020    source источник
comment
Если бы это был сборщик документов Какао, я бы ожидал URL с ограниченным доступом.   -  person matt    schedule 29.09.2020


Ответы (1)


Вы должны сначала вызвать метод startAccessingSecurityScopedResource() для URL-адреса, прежде чем читать содержимое файла. Примечание. не забудьте позвонить stopAccessingSecurityScopedResource() после того, как закончите!

Дополнительную информацию можно найти в документации Apple.

Изменить:

Вот код, который у меня сработал:

.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)
        }
    }
person Cameron    schedule 14.10.2020