Как предоставить локализованное описание с типом ошибки в Swift?

203

Я определяю пользовательский тип ошибки с помощью синтаксиса Swift 3 и хочу предоставить удобное описание ошибки, которая возвращается localizedDescriptionсвойством Errorобъекта. Как мне это сделать?

public enum MyError: Error {
  case customError

  var localizedDescription: String {
    switch self {
    case .customError:
      return NSLocalizedString("A user-friendly description of the error.", comment: "My error")
    }
  }
}

let error: Error = MyError.customError
error.localizedDescription
// "The operation couldn’t be completed. (MyError error 0.)"

Есть ли способ localizedDescriptionвернуть мое пользовательское описание ошибки («Удобное описание ошибки»)? Обратите внимание, что объект ошибки здесь имеет тип, Errorа не MyError. Я могу, конечно, привести объект к MyError

(error as? MyError)?.localizedDescription

но есть ли способ заставить его работать без приведения к моему типу ошибки?

Евгений
источник

Ответы:

403

Как описано в примечаниях к выпуску Xcode 8 beta 6,

Определенные Swift типы ошибок могут предоставить локализованные описания ошибок, приняв новый протокол LocalizedError.

В твоем случае:

public enum MyError: Error {
    case customError
}

extension MyError: LocalizedError {
    public var errorDescription: String? {
        switch self {
        case .customError:
            return NSLocalizedString("A user-friendly description of the error.", comment: "My error")
        }
    }
}

let error: Error = MyError.customError
print(error.localizedDescription) // A user-friendly description of the error.

Вы можете предоставить еще больше информации, если ошибка преобразуется в NSError(что всегда возможно):

extension MyError : LocalizedError {
    public var errorDescription: String? {
        switch self {
        case .customError:
            return NSLocalizedString("I failed.", comment: "")
        }
    }
    public var failureReason: String? {
        switch self {
        case .customError:
            return NSLocalizedString("I don't know why.", comment: "")
        }
    }
    public var recoverySuggestion: String? {
        switch self {
        case .customError:
            return NSLocalizedString("Switch it off and on again.", comment: "")
        }
    }
}

let error = MyError.customError as NSError
print(error.localizedDescription)        // I failed.
print(error.localizedFailureReason)      // Optional("I don\'t know why.")
print(error.localizedRecoverySuggestion) // Optional("Switch it off and on again.")

Принимая CustomNSErrorпротокол, ошибка может обеспечить userInfoсловарь (а также а domainи code). Пример:

extension MyError: CustomNSError {

    public static var errorDomain: String {
        return "myDomain"
    }

    public var errorCode: Int {
        switch self {
        case .customError:
            return 999
        }
    }

    public var errorUserInfo: [String : Any] {
        switch self {
        case .customError:
            return [ "line": 13]
        }
    }
}

let error = MyError.customError as NSError

if let line = error.userInfo["line"] as? Int {
    print("Error in line", line) // Error in line 13
}

print(error.code) // 999
print(error.domain) // myDomain
Мартин Р
источник
7
Есть ли причина , почему вы делаете первый и продлить его позже? Есть ли разница, если вы сделали это в первую очередь? MyErrorErrorLocalizedErrorLocalizedError
Gee.E
9
@ Gee.E: это не имеет значения. Это просто способ организовать код (одно расширение для каждого протокола). Сравните stackoverflow.com/questions/36263892/… , stackoverflow.com/questions/40502086/… или natashatherobot.com/using-swift-extensions .
Мартин Р
4
Ах, проверь. Теперь я понимаю, что вы говорите. Раздел «Соответствие протоколов» на natashatherobot.com/using-swift-extensions - действительно хороший пример того, что вы имеете в виду. Спасибо!
Gee.E
1
@MartinR Если моя ошибка будет преобразована в NSError, как я могу передать словарь из ошибки, к которому можно обратиться как userInfo NSError?
BangOperator
18
Остерегайтесь печатать var errorDescription: String?вместо String. В реализации LocalizedError есть ошибка. Смотри SR-5858 .
ethanhuang13
35

Я также добавил бы, если ваша ошибка имеет такие параметры

enum NetworkError: LocalizedError {
  case responseStatusError(status: Int, message: String)
}

Вы можете назвать эти параметры в вашем локализованном описании следующим образом:

extension NetworkError {
  public var errorDescription: String? {
    switch self {
    case .responseStatusError(status: let status, message: let message):
      return "Error with status \(status) and message \(message) was thrown"
  }
}

Вы можете даже сделать это короче так:

extension NetworkError {
  public var errorDescription: String? {
    switch self {
    case let .responseStatusError(status, message):
      return "Error with status \(status) and message \(message) was thrown"
  }
}
Реза Ширазян
источник
4

В настоящее время существует два протокола принятия ошибок, которые может принять ваш тип ошибки для предоставления дополнительной информации Objective-C - LocalizedError и CustomNSError. Вот пример ошибки, которая принимает оба из них:

enum MyBetterError : CustomNSError, LocalizedError {
    case oops

    // domain
    static var errorDomain : String { return "MyDomain" }
    // code
    var errorCode : Int { return -666 }
    // userInfo
    var errorUserInfo: [String : Any] { return ["Hey":"Ho"] };

    // localizedDescription
    var errorDescription: String? { return "This sucks" }
    // localizedFailureReason
    var failureReason: String? { return "Because it sucks" }
    // localizedRecoverySuggestion
    var recoverySuggestion: String? { return "Give up" }

}
матовый
источник
2
Вы можете сделать редактирование? Ваши примеры не очень помогают понять ценность каждого из них. Или просто удалите его, потому что ответ MartinR предлагает именно это ...
Дорогая,
3

Использование структуры может быть альтернативой. Немного элегантности со статической локализацией:

import Foundation

struct MyError: LocalizedError, Equatable {

   private var description: String!

   init(description: String) {
       self.description = description
   }

   var errorDescription: String? {
       return description
   }

   public static func ==(lhs: MyError, rhs: MyError) -> Bool {
       return lhs.description == rhs.description
   }
}

extension MyError {

   static let noConnection = MyError(description: NSLocalizedString("No internet connection",comment: ""))
   static let requestFailed = MyError(description: NSLocalizedString("Request failed",comment: ""))
}

func throwNoConnectionError() throws {
   throw MyError.noConnection
}

do {
   try throwNoConnectionError()
}
catch let myError as MyError {
   switch myError {
   case .noConnection:
       print("noConnection: \(myError.localizedDescription)")
   case .requestFailed:
       print("requestFailed: \(myError.localizedDescription)")
   default:
      print("default: \(myError.localizedDescription)")
   }
}
Зафер Севик
источник
0

Вот более элегантное решение:

  enum ApiError: String, LocalizedError {

    case invalidCredentials = "Invalid credentials"
    case noConnection = "No connection"

    var localizedDescription: String { return NSLocalizedString(self.rawValue, comment: "") }

  }
Виталий Гоженко
источник
4
Это может быть более элегантно во время выполнения, но на этапе статической локализации не удастся извлечь эти строки для переводчиков; вы увидите "Bad entry in file – Argument is not a literal string"ошибку при запуске exportLocalizationsили genstringsсоздании списка переводимого текста.
Савинола
@savinola согласен, статическая локализация не будет работать в таком случае. Возможно, использовать switch + caseединственный вариант ...
Виталий Гоженко
Использование необработанных значений также предотвратит использование связанных значений для любых ваших ошибок
Броди Робертсон,