This repository has been archived by the owner on Mar 27, 2020. It is now read-only.
forked from Carthage/Carthage
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCompatibilityInfo.swift
75 lines (63 loc) · 3.22 KB
/
CompatibilityInfo.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
import Foundation
import Result
/// Identifies a dependency, its pinned version, and its compatible and incompatible requirements
public struct CompatibilityInfo {
public typealias Requirements = [Dependency: [Dependency: VersionSpecifier]]
/// The dependency
public let dependency: Dependency
/// The pinned version of this dependency
public let pinnedVersion: PinnedVersion
/// Requirements with which the pinned version of this dependency may or may not be compatible
private let requirements: [Dependency: VersionSpecifier]
public init(dependency: Dependency, pinnedVersion: PinnedVersion, requirements: [Dependency: VersionSpecifier]) {
self.dependency = dependency
self.pinnedVersion = pinnedVersion
self.requirements = requirements
}
/// Requirements which are compatible with the pinned version of this dependency
public var compatibleRequirements: [Dependency: VersionSpecifier] {
return requirements.filter { _, version in version.isSatisfied(by: pinnedVersion) }
}
/// Requirements which are not compatible with the pinned version of this dependency
public var incompatibleRequirements: [Dependency: VersionSpecifier] {
return requirements.filter { _, version in !version.isSatisfied(by: pinnedVersion) }
}
/// Accepts a dictionary which maps a dependency to the pinned versions of the dependencies it requires.
/// Returns an inverted dictionary which maps a dependency to the dependencies that require it and the pinned version required
/// e.g. [A: [B: 1, C: 2]] -> [B: [A: 1], C: [A: 2]]
public static func invert(requirements: Requirements) -> Result<Requirements, CarthageError> {
var invertedRequirements: Requirements = [:]
for (dependency, requirements) in requirements {
for (requiredDependency, requiredVersion) in requirements {
var requirements = invertedRequirements[requiredDependency] ?? [:]
if requirements[dependency] != nil {
return .init(error: .duplicateDependencies([DuplicateDependency(dependency: dependency, locations: [])]))
}
requirements[dependency] = requiredVersion
invertedRequirements[requiredDependency] = requirements
}
}
return .init(invertedRequirements)
}
/// Constructs CompatibilityInfo objects for dependencies with incompatibilities
/// given a dictionary of dependencies with pinned versions and their corresponding requirements
public static func incompatibilities(for dependencies: [Dependency: PinnedVersion], requirements: CompatibilityInfo.Requirements) -> Result<[CompatibilityInfo], CarthageError> {
return CompatibilityInfo.invert(requirements: requirements)
.map { invertedRequirements -> [CompatibilityInfo] in
return dependencies.compactMap { dependency, version in
if case .success = SemanticVersion.from(version), let requirements = invertedRequirements[dependency] {
return CompatibilityInfo(dependency: dependency, pinnedVersion: version, requirements: requirements)
}
return nil
}
.filter { !$0.incompatibleRequirements.isEmpty }
}
}
}
extension CompatibilityInfo: Equatable {
public static func == (_ lhs: CompatibilityInfo, _ rhs: CompatibilityInfo) -> Bool {
return lhs.dependency == rhs.dependency &&
lhs.pinnedVersion == rhs.pinnedVersion &&
lhs.requirements == rhs.requirements
}
}