mirror of
https://github.com/nicoverbruggen/phpmon.git
synced 2025-08-08 04:20:07 +02:00
The version constraint checks will also be used in the future to evaluate whether any given site's PHP constraint (if set) is valid for the currently linked version of PHP. For example, assuming you have PHP 8.1.2 linked, we could evaluate: * A site requires "8.0" -> invalid * A site requires "^8.0" -> valid * A site requires "^8.0.0" -> valid * A site requires "~8.0" -> valid * A site requires "~8.0.0" -> invalid Currently, this constraint check is used to determine which versions that are currently installed are good suggestions to switch to. If you have a site with constraint "^8.0" for example, and you have PHP 8.0 and 8.1 installed (with 8.1 linked), then you will get a suggestion to switch back to 8.0.
142 lines
5.9 KiB
Swift
142 lines
5.9 KiB
Swift
//
|
|
// PhpVersionNumber.swift
|
|
// PHP Monitor
|
|
//
|
|
// Created by Nico Verbruggen on 23/01/2022.
|
|
// Copyright © 2022 Nico Verbruggen. All rights reserved.
|
|
//
|
|
|
|
import Foundation
|
|
|
|
public struct PhpVersionNumberCollection: Equatable {
|
|
let versions: [PhpVersionNumber]
|
|
|
|
public static func make(from versions: [String]) -> Self {
|
|
return PhpVersionNumberCollection(
|
|
versions: versions.map { PhpVersionNumber.make(from: $0)! }
|
|
)
|
|
}
|
|
|
|
public var first: PhpVersionNumber? {
|
|
return self.versions.first
|
|
}
|
|
|
|
public var all: [PhpVersionNumber] {
|
|
return self.versions
|
|
}
|
|
|
|
/**
|
|
Checks if any versions of PHP are valid for the constraint provided.
|
|
Due to the complexity of evaluating these, a important test is maintained.
|
|
More information on these constraints can be found here:
|
|
https://getcomposer.org/doc/articles/versions.md#writing-version-constraints
|
|
|
|
- Parameter constraint: The full constraint as a string (e.g. "^7.0")
|
|
- Parameter strict: Whether the minor version check is strict. See more below.
|
|
|
|
The strict mode does not matter if a patch version is provided for all versions in the collection.
|
|
|
|
Strict mode assumes that any PHP version lacking precise patch information, e.g. inferred
|
|
from Homebrew corresponds to the .0 patch version of that version. The default, which is imprecise,
|
|
assumes that the patch version is .999, which means that in all cases the patch version check is
|
|
always going to pass.
|
|
|
|
**STRICT MODE (= patch precision on)**
|
|
|
|
Given versions 8.0.? and 8.1.?, but the requirement is ^8.0.1, in strict mode only 8.1.? will
|
|
be considered valid (8.0 translates to 8.0.0 and as such is older than 8.0.1, 8.1.0 is OK).
|
|
When checking against actual PHP versions installed by the user, use strict mode.
|
|
|
|
**NON-STRICT MODE (= patch precision off)**
|
|
|
|
Given versions 8.0.? and 8.1.?, but the requirement is ^8.0.1, in non-strict mode version 8.0
|
|
is assumed to be equal to version 8.0.999, which is actually fine if 8.0.1 is the required version.
|
|
In non-strict mode, the patch version is ignored for regular version checks (no caret / tilde).
|
|
If checking compatibility with general Homebrew versions of PHP, do NOT use strict mode, since
|
|
the patch version there is not used. (The formula php@8.0 suffices for ^8.0.1.)
|
|
*/
|
|
public func matching(constraint: String, strict: Bool = false) -> [PhpVersionNumber] {
|
|
if let version = PhpVersionNumber.make(from: constraint, type: .versionOnly) {
|
|
// Strict constraint (e.g. "7.0") -> returns specific version
|
|
return self.versions.filter {
|
|
$0.major == version.major
|
|
&& $0.minor == version.minor
|
|
&& (strict ? $0.patch(strict, version) == version.patch(strict) : true)
|
|
}
|
|
}
|
|
|
|
if let version = PhpVersionNumber.make(from: constraint, type: .caretVersionRange) {
|
|
// Caret range means that the major version is never higher but minor version can be higher
|
|
// ^7.2 will be compatible with all versions between 7.2 and 8.0
|
|
return self.versions.filter {
|
|
$0.major == version.major &&
|
|
(
|
|
// Either the minor version is the same and the patch is higher or equal
|
|
$0.minor == version.minor && $0.patch(strict) >= version.patch(strict, $0)
|
|
// or the minor version number has been bumped
|
|
|| $0.minor > version.minor
|
|
)
|
|
}
|
|
}
|
|
|
|
if let version = PhpVersionNumber.make(from: constraint, type: .tildeVersionRange) {
|
|
// Tilde range means that most specific digit is used as the basis.
|
|
if version.patch != nil {
|
|
// If a patch is provided then the minor version cannot be bumped.
|
|
return self.versions.filter {
|
|
$0.major == version.major && $0.minor == version.minor
|
|
&& $0.patch(strict, version) >= version.patch!
|
|
}
|
|
} else {
|
|
// If a patch is not provided then the major version cannot be bumped.
|
|
return self.versions.filter {
|
|
$0.major == version.major && $0.minor >= version.minor
|
|
}
|
|
}
|
|
}
|
|
|
|
return []
|
|
}
|
|
}
|
|
|
|
public struct PhpVersionNumber: Equatable {
|
|
let major: Int
|
|
let minor: Int
|
|
let patch: Int?
|
|
|
|
public func patch(_ strictFallback: Bool, _ constraint: PhpVersionNumber? = nil) -> Int {
|
|
return patch ?? (strictFallback ? 0 : constraint?.patch ?? 999)
|
|
}
|
|
|
|
public var homebrewVersion: String {
|
|
return "\(major).\(minor)"
|
|
}
|
|
|
|
public enum MatchType: String {
|
|
case versionOnly = #"^(?<major>\d+).(?<minor>\d+).?(?<patch>\d+)?\z"#
|
|
case caretVersionRange = #"^\^(?<major>\d+).(?<minor>\d+).?(?<patch>\d+)?\z"#
|
|
case tildeVersionRange = #"^~(?<major>\d+).(?<minor>\d+).?(?<patch>\d+)?\z"#
|
|
}
|
|
|
|
public static func make(from versionString: String, type: MatchType = .versionOnly) -> Self? {
|
|
let regex = try! NSRegularExpression(pattern: type.rawValue, options: [])
|
|
let match = regex.matches(in: versionString, options: [], range: NSMakeRange(0, versionString.count)).first
|
|
|
|
if match != nil {
|
|
let major = Int(
|
|
versionString[Range(match!.range(withName: "major"), in: versionString)!]
|
|
)!
|
|
let minor = Int(
|
|
versionString[Range(match!.range(withName: "minor"), in: versionString)!]
|
|
)!
|
|
var patch: Int? = nil
|
|
if let minorRange = Range(match!.range(withName: "patch"), in: versionString) {
|
|
patch = Int(versionString[minorRange])
|
|
}
|
|
return Self(major: major, minor: minor, patch: patch)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|