All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.openrewrite.maven.tree.Version Maven / Gradle / Ivy

/*
 * Copyright 2020 the original author or authors.
 * 

* Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at *

* https://www.apache.org/licenses/LICENSE-2.0 *

* Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.openrewrite.maven.tree; import org.jspecify.annotations.Nullable; import java.math.BigInteger; import java.util.*; /** * Modified from org.eclipse.aether.util.version.GenericVersion. */ public class Version implements Comparable { private final String version; private final Version.Item[] items; public Version(String version) { this.version = version; this.items = parse(version); } private static Version.Item[] parse(String version) { List items = new ArrayList<>(); Version.Tokenizer tokenizer = new Version.Tokenizer(version); while (tokenizer.next()) { Version.Item item = tokenizer.toItem(); items.add(item); } trimPadding(items); return items.toArray(new Item[0]); } private static void trimPadding(List items) { Boolean number = null; int end = items.size() - 1; for (int i = end; i > 0; --i) { Version.Item item = items.get(i); if (!Boolean.valueOf(item.isNumber()).equals(number)) { end = i; number = item.isNumber(); } if (end == i && (i == items.size() - 1 || items.get(i - 1).isNumber() == item.isNumber()) && item.compareTo(null) == 0) { items.remove(i); --end; } } } @Override public int compareTo(Version obj) { Version.Item[] these = this.items; Version.Item[] those = obj.items; boolean number = true; for (int index = 0; index < these.length || index < those.length; ++index) { if (index >= these.length) { return -comparePadding(those, index, null); } if (index >= those.length) { return comparePadding(these, index, null); } Version.Item thisItem = these[index]; Version.Item thatItem = those[index]; if (thisItem.isNumber() != thatItem.isNumber()) { if (number == thisItem.isNumber()) { return comparePadding(these, index, number); } return -comparePadding(those, index, number); } int rel = thisItem.compareTo(thatItem); if (rel != 0) { return rel; } number = thisItem.isNumber(); } return 0; } private static int comparePadding(Version.Item[] items, int index, @Nullable Boolean number) { int rel = 0; for (int i = index; i < items.length; ++i) { Version.Item item = items[i]; if (number != null && number != item.isNumber()) { break; } rel = item.compareTo(null); if (rel != 0) { break; } } return rel; } @Override public boolean equals(Object obj) { return obj instanceof Version && this.compareTo((Version) obj) == 0; } @Override public int hashCode() { return this.version.hashCode(); } @Override public String toString() { return this.version; } static final class Item { private static final Version.Item MAX = new Version.Item(8, "max"); private static final Version.Item MIN = new Version.Item(0, "min"); private final int kind; private final Object value; Item(int kind, Object value) { this.kind = kind; this.value = value; } public boolean isNumber() { return (this.kind & 2) == 0; } public int compareTo(Version.@Nullable Item that) { int rel; if (that == null) { switch (this.kind) { case 0: rel = -1; break; case 1: case 6: case 7: default: throw new IllegalStateException("unknown version item kind " + this.kind); case 2: case 4: rel = (Integer) this.value; break; case 3: case 5: case 8: rel = 1; } } else { rel = this.kind - that.kind; if (rel == 0) { switch (this.kind) { case 0: case 8: break; case 1: case 6: case 7: default: throw new IllegalStateException("unknown version item kind " + this.kind); case 2: case 4: rel = ((Integer) this.value).compareTo((Integer) that.value); break; case 3: rel = ((String) this.value).compareToIgnoreCase((String) that.value); break; case 5: rel = ((BigInteger) this.value).compareTo((BigInteger) that.value); } } } return rel; } @Override public boolean equals(Object obj) { return obj instanceof Version.Item && this.compareTo((Version.Item) obj) == 0; } @Override public int hashCode() { return this.value.hashCode() + this.kind * 31; } @Override public String toString() { return String.valueOf(this.value); } } static final class Tokenizer { private static final Integer QUALIFIER_ALPHA = -5; private static final Integer QUALIFIER_BETA = -4; private static final Integer QUALIFIER_MILESTONE = -3; private static final Map QUALIFIERS; private final String version; private int index; private String token; private boolean number; private boolean terminatedByNumber; Tokenizer(String version) { this.version = version.length() > 0 ? version : "0"; } public boolean next() { int n = this.version.length(); if (this.index >= n) { return false; } else { int state = -2; int start = this.index; int end = n; for (this.terminatedByNumber = false; this.index < n; ++this.index) { char c = this.version.charAt(this.index); if (c == '.' || c == '-' || c == '_') { end = this.index++; break; } int digit = Character.digit(c, 10); if (digit >= 0) { if (state == -1) { end = this.index; this.terminatedByNumber = true; break; } if (state == 0) { ++start; } state = state <= 0 && digit <= 0 ? 0 : 1; } else { if (state >= 0) { end = this.index; break; } state = -1; } } if (end - start > 0) { this.token = this.version.substring(start, end); this.number = state >= 0; } else { this.token = "0"; this.number = true; } return true; } } @Override public String toString() { return String.valueOf(this.token); } public Version.Item toItem() { if (this.number) { try { return this.token.length() < 10 ? new Version.Item(4, Integer.parseInt(this.token)) : new Version.Item(5, new BigInteger(this.token)); } catch (NumberFormatException var2) { throw new IllegalStateException(var2); } } else { if (this.index >= this.version.length()) { if ("min".equalsIgnoreCase(this.token)) { return Version.Item.MIN; } if ("max".equalsIgnoreCase(this.token)) { return Version.Item.MAX; } } if (this.terminatedByNumber && this.token.length() == 1) { switch (this.token.charAt(0)) { case 'A': case 'a': return new Version.Item(2, QUALIFIER_ALPHA); case 'B': case 'b': return new Version.Item(2, QUALIFIER_BETA); case 'M': case 'm': return new Version.Item(2, QUALIFIER_MILESTONE); } } Integer qualifier = QUALIFIERS.get(this.token); return qualifier != null ? new Version.Item(2, qualifier) : new Version.Item(3, this.token.toLowerCase(Locale.ENGLISH)); } } static { QUALIFIERS = new TreeMap<>(String.CASE_INSENSITIVE_ORDER); QUALIFIERS.put("alpha", QUALIFIER_ALPHA); QUALIFIERS.put("beta", QUALIFIER_BETA); QUALIFIERS.put("milestone", QUALIFIER_MILESTONE); QUALIFIERS.put("cr", -2); QUALIFIERS.put("rc", -2); QUALIFIERS.put("snapshot", -1); QUALIFIERS.put("ga", 0); QUALIFIERS.put("final", 0); QUALIFIERS.put("release", 0); QUALIFIERS.put("", 0); QUALIFIERS.put("sp", 1); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy