diff --git a/hutool-core/src/main/java/cn/hutool/core/util/VersionUtil.java b/hutool-core/src/main/java/cn/hutool/core/util/VersionUtil.java new file mode 100644 index 000000000..6602b7569 --- /dev/null +++ b/hutool-core/src/main/java/cn/hutool/core/util/VersionUtil.java @@ -0,0 +1,196 @@ +package cn.hutool.core.util; + +import cn.hutool.core.collection.CollUtil; +import cn.hutool.core.exceptions.UtilException; + +import java.util.Collection; +import java.util.List; +import java.util.Objects; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * 版本对比工具 + * 对 {@link cn.hutool.core.comparator.VersionComparator} 的封装 + * 最主要功能包括: + * + * + *
+ * 1. 版本表达式匹配
+ * 2. 单个版本匹配
+ * 
+ * + * @author winlans + * @see cn.hutool.core.comparator.VersionComparator + */ +public class VersionUtil { + + private static final Pattern COMPARE_REG = Pattern.compile("^[<>≥≤]=?"); + // 默认多版本分隔符 + private static final String defaultVersionsDelimiter = ";"; + + /** + * 是否匹配任意一个版本 + * + * @param currentVersion 当前版本 + * @param compareVersions 待匹配的版本列表 + * @return true 包含待匹配的版本 + */ + public static boolean anyMatch(String currentVersion, Collection compareVersions) { + return matchEl(currentVersion, CollUtil.join(compareVersions, defaultVersionsDelimiter)); + } + + public static boolean anyMatch(String currentVersion, String... compareVersions) { + return matchEl(currentVersion, ArrayUtil.join(compareVersions, defaultVersionsDelimiter)); + } + + /** + * 当前版本大于待比较版本 + * + * @param currentVersion 当前本本 + * @param compareVersion 待比较版本 + * @return true 当前版本大于待比较版本 + */ + public static boolean isGreaterThan(String currentVersion, String compareVersion) { + return matchEl(currentVersion, ">" + compareVersion); + } + + /** + * 当前版本大于等于待比较版本 + * + * @param currentVersion 当前本本 + * @param compareVersion 待比较版本 + * @return true 当前版本大于等于待比较版本 + */ + public static boolean isGreaterThanOrEqual(String currentVersion, String compareVersion) { + return matchEl(currentVersion, ">=" + compareVersion); + } + + /** + * 当前版本小于待比较版本 + * + * @param currentVersion 当前本本 + * @param compareVersion 待比较版本 + * @return true 当前版本小于待比较版本 + */ + public static boolean isLessThan(String currentVersion, String compareVersion) { + return matchEl(currentVersion, "<" + compareVersion); + } + + /** + * 当前版本小于等于待比较版本 + * + * @param currentVersion 当前本本 + * @param compareVersion 待比较版本 + * @return true 当前版本小于等于待比较版本 + */ + public static boolean isLessThanOrEqual(String currentVersion, String compareVersion) { + return matchEl(currentVersion, "<=" + compareVersion); + } + + /** + * 当前版本是否满足版本表达式 + *
+	 *     matchEl("1.0.2", ">=1.0.2") == true
+	 *     matchEl("1.0.2", "<1.0.1;1.0.2") == true
+	 *     matchEl("1.0.2", "<1.0.2") == false
+	 *     matchEl("1.0.2", "1.0.0-1.1.1") == true
+	 *     matchEl("1.0.2", "1.0.0-1.1.1") == true
+	 * 
+ * + * @param currentVersion 当前本本 + * @param versionEl 版本表达式 + * @return true 当前版本小于等于待比较版本 + */ + public static boolean matchEl(String currentVersion, String versionEl) { + return matchEl(currentVersion, versionEl, defaultVersionsDelimiter); + } + + /** + * 当前版本是否满足版本表达式 + *
+	 *     matchEl("1.0.2", ">=1.0.2", ";") == true
+	 *     matchEl("1.0.2", "<1.0.1,1.0.2", ",") == true
+	 *     matchEl("1.0.2", "<1.0.2", ";") == false
+	 *     matchEl("1.0.2", "1.0.0-1.1.1", ",") == true
+	 *     matchEl("1.0.2", "1.0.1,1.0.2-1.1.1", ",") == true
+	 * 
+ * + * @param currentVersion 当前本本 + * @param versionEl 版本表达式(可以匹配多个条件,使用指定的分隔符(默认;)分隔), + * {@code '-'}表示范围包含左右版本,如果 {@code '-'}的左边没有,表示小于等于某个版本号, 右边表示大于等于某个版本号。 + * 支持比较符号{@code '>'},{@code '<'}, {@code '>='},{@code '<='},{@code '≤'},{@code '≥'} + * + * + * @param versionsDelimiter 多表达式分隔符 + * @return true 当前版本小于等于待比较版本 + */ + public static boolean matchEl(String currentVersion, String versionEl, String versionsDelimiter) { + if (StrUtil.isBlank(versionsDelimiter) + || StrUtil.equals("-", versionsDelimiter) + || ReUtil.isMatch(COMPARE_REG, versionsDelimiter)) { + throw new UtilException("非法的版本分隔符:" + versionsDelimiter); + } + + if (StrUtil.isBlank(versionEl) || StrUtil.isBlank(currentVersion)) { + return false; + } + String trimmedVersion = StrUtil.trim(currentVersion); + + List els = StrUtil.split(versionEl, versionsDelimiter, true, true); + if (CollUtil.isEmpty(els)) { + return false; + } + + for (String el : els) { + el = el.trim(); + Matcher matcher = COMPARE_REG.matcher(el); + if (matcher.find()) { + String op = matcher.group(); + String ver = StrUtil.removePrefix(el, op); + switch (op) { + case ">=": + case "≥": + if (StrUtil.compareVersion(trimmedVersion, ver) >= 0) { + return true; + } + break; + case "<=": + case "≤": + if (StrUtil.compareVersion(trimmedVersion, ver) <= 0) { + return true; + } + break; + case "<": + if (StrUtil.compareVersion(trimmedVersion, ver) < 0) { + return true; + } + break; + case ">": + if (StrUtil.compareVersion(trimmedVersion, ver) > 0) { + return true; + } + break; + default: + return false; + } + } else if (StrUtil.contains(el, "-")) { + String[] pair = el.split("-"); + String left = StrUtil.blankToDefault(StrUtil.trim(pair[0]), ""); + String right = StrUtil.blankToDefault(StrUtil.trim(pair[1]), ""); + + boolean leftMatch = StrUtil.isBlank(left) || StrUtil.compareVersion(left, trimmedVersion) <= 0; + boolean rightMatch = StrUtil.isBlank(right) || StrUtil.compareVersion(right, trimmedVersion) >= 0; + if (leftMatch && rightMatch) { + return true; + } + } else if (Objects.equals(trimmedVersion, el)) { + return true; + } + } + return false; + } +} diff --git a/hutool-core/src/test/java/cn/hutool/core/util/VersionUtilTest.java b/hutool-core/src/test/java/cn/hutool/core/util/VersionUtilTest.java new file mode 100644 index 000000000..4b3dc7893 --- /dev/null +++ b/hutool-core/src/test/java/cn/hutool/core/util/VersionUtilTest.java @@ -0,0 +1,79 @@ +package cn.hutool.core.util; + +import cn.hutool.core.collection.ListUtil; +import cn.hutool.core.exceptions.UtilException; +import org.junit.jupiter.api.Test; + +import java.util.List; + +import static org.junit.jupiter.api.Assertions.*; + +class VersionUtilTest { + + @Test + void isGreaterThan() { + String currentVersion = " 1.0.2"; + assertTrue(VersionUtil.isGreaterThan(currentVersion, "1.0.1")); + assertTrue(VersionUtil.isGreaterThan(currentVersion, "1")); + assertFalse(VersionUtil.isGreaterThan(currentVersion, "1.1")); + } + + @Test + void isGreaterThanOrEqual() { + String currentVersion = "1.0.2 "; + assertTrue(VersionUtil.isGreaterThanOrEqual(currentVersion, "1.0.1")); + assertTrue(VersionUtil.isGreaterThanOrEqual(currentVersion, "1.0.2")); + assertFalse(VersionUtil.isGreaterThanOrEqual(currentVersion, "1.1")); + } + + @Test + void isLessThan() { + String currentVersion = "1.0.2"; + assertTrue(VersionUtil.isLessThan(currentVersion, "1.0.3")); + assertFalse(VersionUtil.isLessThan(currentVersion, "1")); + assertTrue(VersionUtil.isLessThan(currentVersion, "1.1")); + assertFalse(VersionUtil.isLessThan(currentVersion, "1.0.2")); + } + + @Test + void isLessThanOrEqual() { + String currentVersion = "1.0.2"; + assertTrue(VersionUtil.isLessThanOrEqual(currentVersion, "1.0.2")); + assertFalse(VersionUtil.isLessThanOrEqual(currentVersion, "1.0.1")); + assertFalse(VersionUtil.isLessThanOrEqual(currentVersion, "1.1")); + } + + @Test + void matchEl() { + String currentVersion = "1.0.2"; + assertTrue(VersionUtil.matchEl(currentVersion, "1.0.1;1.0.2")); + assertFalse(VersionUtil.matchEl(currentVersion, "1.0.1;1.0.3")); + assertTrue(VersionUtil.matchEl(currentVersion, "1.0.9;1.0.1-1.0.2")); + assertTrue(VersionUtil.matchEl(currentVersion, "1.0.9;1.0.1-1.0.3")); + + assertTrue(VersionUtil.matchEl(currentVersion, "1.0.9,1.0.1-1.0.3", ",")); + } + + @Test + void matchEl_Exception_whenVersionDelimiterIllegal() { + List illegalDelimiters = ListUtil.of("-", ">", ">=", "<", "<=", "≥", "≤", null, "", " "); + + for (String illegalDelimiter : illegalDelimiters) { + assertThrows(UtilException.class, () -> { + String currentVersion = "1.0.2"; + VersionUtil.matchEl(currentVersion, "1.0.1;1.0.2", illegalDelimiter); + }); + } + } + + @Test + void anyMatch() { + String currentVersion = "1.0.2"; + assertTrue(VersionUtil.anyMatch(currentVersion, ListUtil.of("1.0.1", "1.0.3", "1.0.2"))); + assertTrue(VersionUtil.anyMatch(currentVersion, "1.0.1", "1.0.2")); + } + + @Test + void testMatchEl() { + } +}