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

io.spring.initializr.generator.version.VersionParser Maven / Gradle / Ivy

There is a newer version: 0.21.0
Show newest version
/*
 * Copyright 2012-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 io.spring.initializr.generator.version;

import java.util.Collections;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

import io.spring.initializr.generator.version.Version.Qualifier;

import org.springframework.util.Assert;
import org.springframework.util.StringUtils;

/**
 * Parser for {@link Version} and {@link VersionRange} that allows to resolve the minor
 * and patch value against a configurable list of "latest versions".
 * 

* For example a parser that is configured with {@code 1.3.7.RELEASE} and * {@code 1.4.2.RELEASE} as latest versions can parse {@code 1.3.x.RELEASE} to * {@code 1.3.7.RELEASE}. Note that the qualifier is important here: * {@code 1.3.8.BUILD-SNAPSHOT} would be parsed as {@code 1.3.999.BUILD-SNAPSHOT} as the * parser doesn't know the latest {@code BUILD-SNAPSHOT} in the {@code 1.3.x} release * line. * * @author Stephane Nicoll */ public class VersionParser { /** * The default {@link VersionParser}. */ public static final VersionParser DEFAULT = new VersionParser(Collections.emptyList()); private static final Pattern VERSION_REGEX = Pattern .compile("^(\\d+)\\.(\\d+|x)\\.(\\d+|x)(?:([.|-])([^0-9]+)(\\d+)?)?$"); private static final Pattern RANGE_REGEX = Pattern.compile("(\\(|\\[)(.*),(.*)(\\)|\\])"); private final List latestVersions; public VersionParser(List latestVersions) { this.latestVersions = latestVersions; } /** * Parse the string representation of a {@link Version}. Throws an * {@link InvalidVersionException} if the version could not be parsed. * @param text the version text * @return a Version instance for the specified version text * @throws InvalidVersionException if the version text could not be parsed * @see #safeParse(java.lang.String) */ public Version parse(String text) { Assert.notNull(text, "Text must not be null"); Matcher matcher = VERSION_REGEX.matcher(text.trim()); if (!matcher.matches()) { throw new InvalidVersionException("Could not determine version based on '" + text + "': version format " + "is Major.Minor.Patch and an optional Qualifier " + "(e.g. 1.0.5.RELEASE)"); } Integer major = Integer.valueOf(matcher.group(1)); String minor = matcher.group(2); String patch = matcher.group(3); Qualifier qualifier = parseQualifier(matcher); if ("x".equals(minor) || "x".equals(patch)) { Integer minorInt = ("x".equals(minor) ? null : Integer.parseInt(minor)); Version latest = findLatestVersion(major, minorInt, qualifier); if (latest == null) { return new Version(major, ("x".equals(minor) ? 999 : Integer.parseInt(minor)), ("x".equals(patch) ? 999 : Integer.parseInt(patch)), qualifier); } return new Version(major, latest.getMinor(), latest.getPatch(), latest.getQualifier()); } else { return new Version(major, Integer.parseInt(minor), Integer.parseInt(patch), qualifier); } } private Qualifier parseQualifier(Matcher matcher) { String qualifierSeparator = matcher.group(4); String qualifierId = matcher.group(5); if (StringUtils.hasText(qualifierSeparator) && StringUtils.hasText(qualifierId)) { String versionString = matcher.group(6); return new Qualifier(qualifierId, (versionString != null) ? Integer.valueOf(versionString) : null, qualifierSeparator); } return null; } /** * Parse safely the specified string representation of a {@link Version}. *

* Return {@code null} if the text represents an invalid version. * @param text the version text * @return a Version instance for the specified version text * @see #parse(java.lang.String) */ public Version safeParse(String text) { try { return parse(text); } catch (InvalidVersionException ex) { return null; } } /** * Parse the string representation of a {@link VersionRange}. Throws an * {@link InvalidVersionException} if the range could not be parsed. * @param text the range text * @return a VersionRange instance for the specified range text * @throws InvalidVersionException if the range text could not be parsed */ public VersionRange parseRange(String text) { Assert.notNull(text, "Text must not be null"); Matcher matcher = RANGE_REGEX.matcher(text.trim()); if (!matcher.matches()) { // Try to read it as simple string Version version = parse(text); return new VersionRange(version, true, null, true); } boolean lowerInclusive = matcher.group(1).equals("["); Version lowerVersion = parse(matcher.group(2)); Version higherVersion = parse(matcher.group(3)); boolean higherInclusive = matcher.group(4).equals("]"); return new VersionRange(lowerVersion, lowerInclusive, higherVersion, higherInclusive); } private Version findLatestVersion(Integer major, Integer minor, Version.Qualifier qualifier) { List matches = this.latestVersions.stream().filter((it) -> { if (major != null && !major.equals(it.getMajor())) { return false; } if (minor != null && !minor.equals(it.getMinor())) { return false; } if (qualifier != null && !qualifier.equals(it.getQualifier())) { return false; } return true; }).collect(Collectors.toList()); return (matches.size() != 1) ? null : matches.get(0); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy