Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/alexruperez/SecurePropertyStorage
Helps you define secure storages for your properties using Swift property wrappers.
https://github.com/alexruperez/SecurePropertyStorage
dependency-injection keychain property-wrappers singleton swift userdefaults
Last synced: 3 months ago
JSON representation
Helps you define secure storages for your properties using Swift property wrappers.
- Host: GitHub
- URL: https://github.com/alexruperez/SecurePropertyStorage
- Owner: alexruperez
- License: mit
- Created: 2020-03-22T20:47:12.000Z (over 4 years ago)
- Default Branch: master
- Last Pushed: 2024-05-25T03:52:32.000Z (5 months ago)
- Last Synced: 2024-05-28T05:02:59.997Z (5 months ago)
- Topics: dependency-injection, keychain, property-wrappers, singleton, swift, userdefaults
- Language: Swift
- Homepage: https://alexruperez.github.io/SecurePropertyStorage
- Size: 31 MB
- Stars: 471
- Watchers: 13
- Forks: 20
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
- Codeowners: .github/CODEOWNERS
Awesome Lists containing this project
- awesome-ios - SecurePropertyStorage - Helps you define secure storages for your properties using Swift property wrappers. (Security / Unofficial)
- awesome-swift - SecurePropertyStorage - Helps you define secure storages for your properties using Swift property wrappers. (Libs / Security)
- awesome-ios-star - SecurePropertyStorage - Helps you define secure storages for your properties using Swift property wrappers. (Security / Unofficial)
- fucking-awesome-swift - SecurePropertyStorage - Helps you define secure storages for your properties using Swift property wrappers. (Libs / Security)
- awesome-swift - SecurePropertyStorage - Helps you define secure storages for your properties using Swift property wrappers. ` ๐ 2 months ago ` (Security [๐](#readme))
README
# ๐ Secure Property Storage
> Helps you define secure storages for your properties using Swift *property wrappers*.[![Twitter](https://img.shields.io/badge/contact-%40alexruperez-blue)](http://alexruperez.com)
[![Swift](https://img.shields.io/badge/swift-5-orange)](https://swift.org)
[![License](https://img.shields.io/github/license/alexruperez/SecurePropertyStorage)](LICENSE)
[![Swift Package Manager](https://img.shields.io/badge/Swift%20Package%20Manager-compatible-4BC51D.svg?style=flat)](https://swift.org/package-manager)
[![Carthage](https://img.shields.io/badge/Carthage-compatible-4BC51D.svg?style=flat)](https://github.com/Carthage/Carthage)
[![Bitrise](https://app.bitrise.io/app/4fed1af31836d3bc/status.svg?token=bYImtoKj0hxqCxnORhdgyg&branch=master)](https://app.bitrise.io/app/4fed1af31836d3bc)
[![Maintainability](https://api.codeclimate.com/v1/badges/bbf38ddca9a26703cefd/maintainability)](https://codeclimate.com/github/alexruperez/SecurePropertyStorage/maintainability)
[![Test Coverage](https://api.codeclimate.com/v1/badges/bbf38ddca9a26703cefd/test_coverage)](https://codeclimate.com/github/alexruperez/SecurePropertyStorage/test_coverage)
[![codecov](https://codecov.io/gh/alexruperez/SecurePropertyStorage/graph/badge.svg?token=T3NNtLnmsA)](https://codecov.io/gh/alexruperez/SecurePropertyStorage)
[![codebeat badge](https://codebeat.co/badges/ee2372f0-2188-43c6-bf7b-f6860834096c)](https://codebeat.co/projects/github-com-alexruperez-securepropertystorage-master)
[![Quality](https://api.codacy.com/project/badge/Grade/53a23dd2feca4b7ca8357c918c7d49c9)](https://app.codacy.com/gh/alexruperez/SecurePropertyStorage/dashboard)
[![Documentation](https://alexruperez.github.io/SecurePropertyStorage/badge.svg)](https://alexruperez.github.io/SecurePropertyStorage)## ๐ Features
All keys are hashed using [SHA512](https://en.wikipedia.org/wiki/SHA-2) and all values are encrypted using [AES](https://en.wikipedia.org/wiki/Advanced_Encryption_Standard)-[GCM](https://en.wikipedia.org/wiki/Galois/Counter_Mode) to keep user information safe, auto*magic*ally.
Symmetric key is stored in Keychain in a totally secure way.## ๐ Basic usage
### @UserDefault
This property wrapper will store your property in [UserDefaults](https://developer.apple.com/documentation/foundation/userdefaults) using `StoreKey` (any `String` but i recommend you a String typed enum).
Optionally, you can assign a default value to the property that will be secure stored at initialization.```swift
@UserDefault(<#StoreKey#>)
var yourProperty: YourType? = yourDefaultValueIfNeeded
```[`UserDefaultsStorage`](Sources/UserDefault/UserDefaultsStorage.swift) is also available, a subclass of [`UserDefaults`](https://developer.apple.com/documentation/foundation/userdefaults) with all the security provided by this library, where you can customize suite name.
### @Keychain
This property wrapper will store your property in [Keychain](https://developer.apple.com/documentation/security/keychain_services) using `StoreKey`.
```swift
@Keychain(<#StoreKey#>)
var yourProperty: YourType? = yourDefaultValueIfNeeded
```As `UserDefaultsStorage`, [`KeychainStorage`](Sources/Keychain/KeychainStorage.swift) is also available, where you can customize access, group and synchronize it with iCloud.
### @Singleton
This property wrapper will store your property in a memory [singleton](https://en.wikipedia.org/wiki/Singleton_pattern), every property with the same wrapper and key can access or modify the value from wherever it is.
```swift
@Singleton(<#StoreKey#>)
var yourProperty: YourType? = yourDefaultValueIfNeeded
```As `KeychainStorage`, [`SingletonStorage`](Sources/Singleton/SingletonStorage.swift) is also available.
### @Inject
This property wrapper is similar to `@Singleton` but, together with `@Register`, will inject your dependencies. More details in [Dependency Injection usage](#-dependency-injection-usage) guide.
```swift
@Inject
var yourDependency: YourProtocol?
```As `SingletonStorage`, [`InjectStorage`](Sources/Inject/InjectStorage.swift) is also available.
### @Store
This is a custom wrapper, you can define your own [`Storage`](Sources/Storage/Storage.swift) protocol implementation.
```swift
@Store(<#YourStorage#>, <#StoreKey#>)
var yourProperty: YourType? = yourDefaultValueIfNeeded
```As `InjectStorage`, [`DelegatedStorage`](Sources/Storage/DelegatedStorage.swift) is also available with all the magic of this library.
## ๐งโโ๏ธ Codable usage
If your property conforms [`Codable`](https://developer.apple.com/documentation/swift/codable) protocol, just add `Codable` keyword as prefix of your property wrapper.
- **@CodableUserDefault**
- **@CodableKeychain**
- **@CodableSingleton**
- **@CodableStore**## ๐ฅก Unwrapped usage
To avoid continually unwrapping your property, just add `Unwrapped` keyword as prefix of your property wrapper, assign a default value (mandatory except for `@UnwrappedInject`), and it will return stored value or default value, but your property will always be there for you.
- **@UnwrappedUserDefault**
- **@UnwrappedKeychain**
- **@UnwrappedSingleton**
- **@UnwrappedInject**
- **@UnwrappedStore**## ๐ฅก + ๐งโโ๏ธ Combo usage
You can also combine previous cases in case you need it, unwrapped first please.
- **@UnwrappedCodableUserDefault**
- **@UnwrappedCodableKeychain**
- **@UnwrappedCodableSingleton**
- **@UnwrappedCodableStore**## ๐ Dependency Injection usage
@Register (click to expand)
This property wrapper will register the implementations of your dependencies.
Register them wherever you want before inject it, but be sure to do it only once (except if you use qualifiers), for example, in an `Injector` class.
You can register through a protocol or directly using your class implementation.```swift
@Register
var yourDependency: YourProtocol = YourImplementation()@Register
var yourDependency = YourImplementation()
```You can also define a closure that builds your dependency.
Just remember cast your dependency if you are going to inject it through a protocol.```swift
@Register
var yourDependency = {
YourImplementation() as YourProtocol
}@Register
var yourDependency = {
YourImplementation()
}
```You can also register your dependencies only after the moment someone tries to inject them and you haven't registered them yet,
for this you can use the error closure.```swift
InjectStorage.standard.errorClosure = { error in
if case InjectError.notFound = error {
YourImplementation.register()
}
}
```You can get this syntactic sugar because you can now use property wrappers in function parameters.
```swift
static func register(@Register yourDependency: YourProtocol = YourImplementation()) {}
```@Inject and @UnwrappedInject (click to expand)
These property wrappers injects your dependencies `@Register` implementations.
```swift
@Inject
var yourDependency: YourProtocol?@Inject
var yourDependency: YourImplementation?@UnwrappedInject
var yourUnwrappedDependency: YourProtocol@UnwrappedInject
var yourUnwrappedDependency: YourImplementation
```#### Scope
Because these property wrappers works similarly to `@Singleton`, the default scope is `.singleton`, but if you use builder closures on `@Register`, you can modify them to inject a single instance.
```swift
@Inject(.instance)
var yourDependency: YourProtocol?@UnwrappedInject(.instance)
var yourUnwrappedDependency: YourProtocol
```@InjectWith and @UnwrappedInjectWith (click to expand)
Your dependency may need parameters when injecting, you can pass them with these property wrappers.
Simply define a model with your dependency parameters and pass it.
It will inject a new instance built with these parameters.```swift
@Register
var yourDependency = { parameters in
YourImplementation(parameters) as YourProtocol
}@Inject(YourParameters())
var yourDependency: YourProtocol?@UnwrappedInject(YourParameters())
var yourUnwrappedDependency: YourProtocol
```Qualifiers (click to expand)
You can use [qualifiers](https://javaee.github.io/tutorial/cdi-basic006.html) to provide various implementations of a particular dependency. A qualifier is just a `@objc protocol` that you apply to a `class`.
For example, you could declare `Dog` and `Cat` qualifier protocols and apply it to another class that conforms `Animal` protocol. To declare this qualifier, use the following code:
```swift
protocol Animal {
func sound()
}@objc protocol Dog {}
@objc protocol Cat {}
```You can then define multiple classes that conforms `Animal` protocol and uses this qualifiers:
```swift
class DogImplementation: Animal, Dog {
func sound() { print("Woof!") }
}class CatImplementation: Animal, Cat {
func sound() { print("Meow!") }
}
```Both implementations of the class can now be `@Register`:
```swift
@Register
var registerDog: Animal = DogImplementation()@Register
var registerCat: Animal = CatImplementation()
```To inject one or the other implementation, simply add the qualifier(s) to your `@Inject`:
```swift
@UnwrappedInject(Dog.self)
var dog: Animal@UnwrappedInject(Cat.self)
var cat: Animaldog.sound() // prints Woof!
cat.sound() // prints Meow!
```Testing (click to expand)
One of the advantages of dependency injection is that the code can be easily testable with mock implementation.
That is why there is a `Mock` qualifier that has priority over all, so you can have your dependencies defined in the app and create your mock in the test target simply by adding this qualifier.```swift
// App targetclass YourImplementation: YourProtocol {}
@Register
var yourDependency: YourProtocol = YourImplementation()@Inject
var yourDependency: YourProtocol?
``````swift
// Test targetclass YourMock: YourProtocol, Mock {}
@Register
var yourDependency: YourProtocol = YourMock()
```Groups (click to expand)
When you have **a lot** of dependencies in your app, you may want to optimize dependency resolution.
You can group them using `@Register(group:)` and a `DependencyGroupKey`:
```swift
@Register(group: <#DependencyGroupKey#>)
var yourDependency: YourProtocol = YourImplementation()
````@Inject(group:)` will look for those dependencies only in that group:
```swift
@Inject(group: <#DependencyGroupKey#>)
var yourDependency: YourProtocol?
```## ๐ Examples
> Talk is cheap. Show me the code.
```swift
// Securely stored in UserDefaults.
@UserDefault("username")
var username: String?// Securely stored in Keychain.
@Keychain("password")
var password: String?// Securely stored in a Singleton storage.
@Singleton("sessionToken")
var sessionToken: String?// Securely stored in a Singleton storage.
// Always has a value, the stored or the default.
@UnwrappedSingleton("refreshToken")
var refreshToken: String = "B0610306-A33F"struct User: Codable {
let username: String
let password: String?
let sessionToken: String?
}// Codable model securely stored in UserDefaults.
@CodableUserDefault("user")
var user: User?
```## ๐ Compatibility
- macOS 10.15+
- iOS 13.0+
- iPadOS 13.0+
- tvOS 13.0+
- watchOS 6.0+
- visionOS 1.0+## โ๏ธ Installation
#### You can use the [Swift Package Manager](https://github.com/apple/swift-package-manager) by declaring SecurePropertyStorage as a dependency in your `Package.swift` file:
```swift
.package(url: "https://github.com/alexruperez/SecurePropertyStorage", from: "0.7.1")
```By default, all property wrappers are installed and you can `import` them, but if you want, you can install only some of them:
- **UserDefault**: @*UserDefault property wrappers.
- **Keychain**: @*Keychain property wrappers.
- **Singleton**: @*Singleton property wrappers.
- **Storage**: @*Store property wrappers.
- **Inject**: @*Inject property wrappers.*For more information, see [the Swift Package Manager documentation](https://github.com/apple/swift-package-manager/tree/master/Documentation).*
#### Or you can use [Carthage](https://github.com/Carthage/Carthage):
```ogdl
github "alexruperez/SecurePropertyStorage"
```## ๐ป Etc.
- Featured in [Dave Verwer](https://twitter.com/daveverwer)'s iOS Dev Weekly - [Issue 450](https://iosdevweekly.com/issues/450?#ll98q5s), thanks Dave!
- Contributions are very welcome. Thanks [Alberto Garcia](https://github.com/AlbGarciam) and [Chen](https://github.com/qchenqizhi)!
- Attribution is appreciated (let's spread the word!), but not mandatory.## ๐จโ๐ป Author
Alex Rupรฉrez โ [@alexruperez](https://twitter.com/alexruperez) โ [email protected]
## ๐ฎโโ๏ธ License
*SecurePropertyStorage* is available under the MIT license. See the [LICENSE](LICENSE) file for more info.