-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathEnvironmentValues.swift
161 lines (143 loc) · 4.17 KB
/
EnvironmentValues.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
//
// EnvironmentValues.swift
// Macro
//
// Created by Helge Hess.
// Copyright © 2020 ZeeZide GmbH. All rights reserved.
//
/**
* A protocol describing a value which can be put into a `Macro` environment
* (`EnvironmentValue`'s).
* The sole purpose is to avoid stringly typed environment keys (like
* "")
*
* Example:
*
* enum LoginUserEnvironmentKey: EnvironmentKey {
* static let defaultValue = ""
* }
*
* In addition to the key definition, one usually declares an accessor to the
* respective environment holder, for example the `IncomingMessage`:
*
* extension IncomingMessage {
*
* var loginUser : String {
* set { self[LoginUserEnvironmentKey.self] = newValue }
* get { self[LoginUserEnvironmentKey.self] }
* }
* }
*
* It can then be used like:
*
* app.use { req, res, next in
* console.log("active user:", req.loginUser)
* next()
* }
*
* If the value really is optional, it can be declared an optional:
*
* enum DatabaseConnectionEnvironmentKey: EnvironmentKey {
* static let defaultValue : DatabaseConnection?
* }
*
* To add a shorter name for environment dumps, implement the `loggingKey`
* property:
*
* enum DatabaseConnectionEnvironmentKey: EnvironmentKey {
* static let defaultValue : DatabaseConnection? = nil
* static let loggingKey = "db"
* }
*
*/
public protocol EnvironmentKey {
associatedtype Value
/**
* If a value isn't set in the environment, the `defaultValue` will be
* returned.
*/
static var defaultValue: Self.Value { get }
/**
* The logging key is used when the environment is logged into a string.
* (It defaults to the Swift runtime name of the implementing type).
*/
static var loggingKey : String { get }
}
public extension EnvironmentKey {
@inlinable
static var loggingKey : String {
return String(describing: self)
}
}
#if swift(>=5.1)
/**
* A dictionary which can hold values assigned to `EnvironmentKey`s.
*
* This is a way to avoid stringly typed `extra` APIs by making use of type
* identity in Swift.
* In Node/JS you would usually just attach properties to the `IncomingMessage`
* object.
*
* To drive `EnvironmentValues`, `EnvironmentKey`s need to be defined. Since
* the type of an `EnviromentKey` is globally unique within Swift, it can be
* used to key into the structure to the store the associated value.
*
* Note that in APIs like SwiftUI or SwiftBlocksUI, EnvironmentValues are
* usually "stacked" in the hierarchy of Views or Blocks.
* That's not necessarily the case in Macro, though Macro application can also
* use it like that.
*/
@frozen
public struct EnvironmentValues {
@usableFromInline
var values = [ ObjectIdentifier : ( loggingKey: String, value: Any ) ]()
}
#else
public struct EnvironmentValues { // 5.0 compat, no @frozen
@usableFromInline
var values = [ ObjectIdentifier : ( loggingKey: String, value: Any ) ]()
}
#endif
public extension EnvironmentValues {
static let empty = EnvironmentValues()
@inlinable
var isEmpty : Bool { return values.isEmpty }
@inlinable
var count : Int { return values.count }
@inlinable
subscript<K: EnvironmentKey>(key: K.Type) -> K.Value {
set {
values[ObjectIdentifier(key)] = ( K.loggingKey, newValue )
}
get {
guard let value = values[ObjectIdentifier(key)]?.value else {
return K.defaultValue
}
guard let typedValue = value as? K.Value else {
assertionFailure("unexpected typed value: \(value)")
return K.defaultValue
}
return typedValue
}
}
@inlinable
var loggingDictionary : [ String : Any ] {
var dict = [ String : Any ]()
dict.reserveCapacity(values.count)
for ( key, value ) in values.values {
dict[key] = value
}
return dict
}
}
public protocol EnvironmentValuesHolder {
var environment : EnvironmentValues { set get }
subscript<K: EnvironmentKey>(key: K.Type) -> K.Value { set get }
}
public extension EnvironmentValuesHolder {
@inlinable
subscript<K: EnvironmentKey>(key: K.Type) -> K.Value {
set { environment[key] = newValue }
get { return environment[key] }
}
}