Refactor VersionParser to improve error handling

This commit is contained in:
Zafar Khaja 2014-06-23 21:32:16 +03:00
parent 82dd3c2968
commit d36d961555
3 changed files with 182 additions and 19 deletions

View File

@ -139,6 +139,8 @@ public class Version implements Comparable<Version> {
* Builds a {@code Version} object.
*
* @return a newly built {@code Version} instance
* @throws ParseException when invalid version string is provided
* @throws UnexpectedCharacterException is a special case of {@code ParseException}
*/
public Version build() {
StringBuilder sb = new StringBuilder();
@ -254,6 +256,8 @@ public class Version implements Comparable<Version> {
* @param version the version string to parse
* @return a new instance of the {@code Version} class
* @throws IllegalArgumentException if the input string is {@code NULL} or empty
* @throws ParseException when invalid version string is provided
* @throws UnexpectedCharacterException is a special case of {@code ParseException}
*/
public static Version valueOf(String version) {
return VersionParser.parseValidSemVer(version);
@ -328,6 +332,8 @@ public class Version implements Comparable<Version> {
* @param preRelease the pre-release version to append
* @return a new instance of the {@code Version} class
* @throws IllegalArgumentException if the input string is {@code NULL} or empty
* @throws ParseException when invalid version string is provided
* @throws UnexpectedCharacterException is a special case of {@code ParseException}
*/
public Version incrementMajorVersion(String preRelease) {
return new Version(
@ -351,6 +357,8 @@ public class Version implements Comparable<Version> {
* @param preRelease the pre-release version to append
* @return a new instance of the {@code Version} class
* @throws IllegalArgumentException if the input string is {@code NULL} or empty
* @throws ParseException when invalid version string is provided
* @throws UnexpectedCharacterException is a special case of {@code ParseException}
*/
public Version incrementMinorVersion(String preRelease) {
return new Version(
@ -374,6 +382,8 @@ public class Version implements Comparable<Version> {
* @param preRelease the pre-release version to append
* @return a new instance of the {@code Version} class
* @throws IllegalArgumentException if the input string is {@code NULL} or empty
* @throws ParseException when invalid version string is provided
* @throws UnexpectedCharacterException is a special case of {@code ParseException}
*/
public Version incrementPatchVersion(String preRelease) {
return new Version(
@ -406,6 +416,8 @@ public class Version implements Comparable<Version> {
* @param preRelease the pre-release version to set
* @return a new instance of the {@code Version} class
* @throws IllegalArgumentException if the input string is {@code NULL} or empty
* @throws ParseException when invalid version string is provided
* @throws UnexpectedCharacterException is a special case of {@code ParseException}
*/
public Version setPreReleaseVersion(String preRelease) {
return new Version(normal, VersionParser.parsePreRelease(preRelease));
@ -417,6 +429,8 @@ public class Version implements Comparable<Version> {
* @param build the build metadata to set
* @return a new instance of the {@code Version} class
* @throws IllegalArgumentException if the input string is {@code NULL} or empty
* @throws ParseException when invalid version string is provided
* @throws UnexpectedCharacterException is a special case of {@code ParseException}
*/
public Version setBuildMetadata(String build) {
return new Version(normal, preRelease, VersionParser.parseBuild(build));

View File

@ -257,15 +257,19 @@ class VersionParser implements Parser<Version> {
private Version parseValidSemVer() {
NormalVersion normal = parseVersionCore();
MetadataVersion preRelease = MetadataVersion.NULL;
if (chars.positiveLookahead(HYPHEN)) {
chars.consume();
preRelease = parsePreRelease();
}
MetadataVersion build = MetadataVersion.NULL;
if (chars.positiveLookahead(PLUS)) {
chars.consume();
Character next = consumeNextCharacter(HYPHEN, PLUS, EOL);
if (HYPHEN.isMatchedBy(next)) {
preRelease = parsePreRelease();
next = consumeNextCharacter(PLUS, EOL);
if (PLUS.isMatchedBy(next)) {
build = parseBuild();
}
} else if (PLUS.isMatchedBy(next)) {
build = parseBuild();
}
consumeNextCharacter(EOL);
return new Version(normal, preRelease, build);
}
@ -282,9 +286,9 @@ class VersionParser implements Parser<Version> {
*/
private NormalVersion parseVersionCore() {
int major = Integer.parseInt(numericIdentifier());
chars.consume(DOT);
consumeNextCharacter(DOT);
int minor = Integer.parseInt(numericIdentifier());
chars.consume(DOT);
consumeNextCharacter(DOT);
int patch = Integer.parseInt(numericIdentifier());
return new NormalVersion(major, minor, patch);
}
@ -305,14 +309,16 @@ class VersionParser implements Parser<Version> {
* @throws ParseException if the pre-release version has empty identifier(s)
*/
private MetadataVersion parsePreRelease() {
ensureValidLookahead(DIGIT, LETTER, HYPHEN);
List<String> idents = new ArrayList<String>();
do {
checkForEmptyIdentifier();
idents.add(preReleaseIdentifier());
if (chars.positiveLookahead(DOT)) {
chars.consume(DOT);
consumeNextCharacter(DOT);
continue;
}
} while (!chars.positiveLookahead(PLUS, EOL));
break;
} while (true);
return new MetadataVersion(idents.toArray(new String[idents.size()]));
}
@ -329,6 +335,7 @@ class VersionParser implements Parser<Version> {
* @return a single pre-release identifier
*/
private String preReleaseIdentifier() {
checkForEmptyIdentifier();
CharType boundary = nearestCharType(DOT, PLUS, EOL);
if (chars.positiveLookaheadBefore(boundary, LETTER, HYPHEN)) {
return alphanumericIdentifier();
@ -353,14 +360,16 @@ class VersionParser implements Parser<Version> {
* @throws ParseException if the build metadata has empty identifier(s)
*/
private MetadataVersion parseBuild() {
ensureValidLookahead(DIGIT, LETTER, HYPHEN);
List<String> idents = new ArrayList<String>();
do {
checkForEmptyIdentifier();
idents.add(buildIdentifier());
if (chars.positiveLookahead(DOT)) {
chars.consume(DOT);
consumeNextCharacter(DOT);
continue;
}
} while (!chars.positiveLookahead(EOL));
break;
} while (true);
return new MetadataVersion(idents.toArray(new String[idents.size()]));
}
@ -377,6 +386,7 @@ class VersionParser implements Parser<Version> {
* @return a single build identifier
*/
private String buildIdentifier() {
checkForEmptyIdentifier();
CharType boundary = nearestCharType(DOT, EOL);
if (chars.positiveLookaheadBefore(boundary, LETTER, HYPHEN)) {
return alphanumericIdentifier();
@ -421,7 +431,7 @@ class VersionParser implements Parser<Version> {
private String alphanumericIdentifier() {
StringBuilder sb = new StringBuilder();
do {
sb.append(chars.consume(DIGIT, LETTER, HYPHEN));
sb.append(consumeNextCharacter(DIGIT, LETTER, HYPHEN));
} while (chars.positiveLookahead(DIGIT, LETTER, HYPHEN));
return sb.toString();
}
@ -441,7 +451,7 @@ class VersionParser implements Parser<Version> {
private String digits() {
StringBuilder sb = new StringBuilder();
do {
sb.append(chars.consume(DIGIT));
sb.append(consumeNextCharacter(DIGIT));
} while (chars.positiveLookahead(DIGIT));
return sb.toString();
}
@ -471,7 +481,7 @@ class VersionParser implements Parser<Version> {
private void checkForLeadingZeroes() {
Character la1 = chars.lookahead(1);
Character la2 = chars.lookahead(2);
if (la1 == '0' && DIGIT.isMatchedBy(la2)) {
if (la1 != null && la1 == '0' && DIGIT.isMatchedBy(la2)) {
throw new ParseException(
"Numeric identifier MUST NOT contain leading zeroes"
);
@ -485,8 +495,39 @@ class VersionParser implements Parser<Version> {
* metadata have empty identifier(s)
*/
private void checkForEmptyIdentifier() {
if (DOT.isMatchedBy(chars.lookahead(1))) {
throw new ParseException("Identifiers MUST NOT be empty");
Character la = chars.lookahead(1);
if (DOT.isMatchedBy(la) || PLUS.isMatchedBy(la) || EOL.isMatchedBy(la)) {
throw new ParseException(
"Identifiers MUST NOT be empty",
new UnexpectedCharacterException(la, DIGIT, LETTER, HYPHEN)
);
}
}
/**
* Tries to consume the next character in the stream.
*
* @param expected the expected types of the next character
* @return the next character in the stream
* @throws UnexpectedCharacterException if the next element is of an unexpected type
*/
private Character consumeNextCharacter(CharType... expected) {
try {
return chars.consume(expected);
} catch (UnexpectedElementException e) {
throw new UnexpectedCharacterException(e);
}
}
/**
* Checks if the next character in the stream is valid.
*
* @param expected the expected types of the next character
* @throws UnexpectedCharacterException if the next element is not valid
*/
private void ensureValidLookahead(CharType... expected) {
if (!chars.positiveLookahead(expected)) {
throw new UnexpectedCharacterException(chars.lookahead(1), expected);
}
}
}

View File

@ -0,0 +1,108 @@
/*
* The MIT License
*
* Copyright 2014 Zafar Khaja <zafarkhaja@gmail.com>.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package com.github.zafarkhaja.semver;
import com.github.zafarkhaja.semver.VersionParser.CharType;
import java.util.Arrays;
import java.util.Collection;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameters;
import static com.github.zafarkhaja.semver.VersionParser.CharType.*;
import static org.junit.Assert.*;
/**
*
* @author Zafar Khaja <zafarkhaja@gmail.com>
*/
@RunWith(Parameterized.class)
public class ParserErrorHandlingTest {
private final String invalidVersion;
private final Character unexpected;
private final CharType[] expected;
public ParserErrorHandlingTest(
String invalidVersion,
Character unexpected,
CharType[] expected
) {
this.invalidVersion = invalidVersion;
this.unexpected = unexpected;
this.expected = expected;
}
@Test
public void shouldCorrectlyHandleParseErrors() {
try {
VersionParser.parseValidSemVer(invalidVersion);
} catch (UnexpectedCharacterException e) {
assertEquals(unexpected, e.getUnexpectedCharacter());
assertArrayEquals(expected, e.getExpectedCharTypes());
return;
} catch (ParseException e) {
if (e.getCause() != null) {
UnexpectedCharacterException cause = (UnexpectedCharacterException) e.getCause();
assertEquals(unexpected, cause.getUnexpectedCharacter());
assertArrayEquals(expected, cause.getExpectedCharTypes());
}
return;
}
fail("Uncaught exception");
}
@Parameters(name = "{0}")
public static Collection<Object[]> parameters() {
return Arrays.asList(new Object[][] {
{ "1", null, new CharType[] { DOT } },
{ "1 ", ' ', new CharType[] { DOT } },
{ "1.", null, new CharType[] { DIGIT } },
{ "1.2", null, new CharType[] { DOT } },
{ "1.2.", null, new CharType[] { DIGIT } },
{ "a.b.c", 'a', new CharType[] { DIGIT } },
{ "1.b.c", 'b', new CharType[] { DIGIT } },
{ "1.2.c", 'c', new CharType[] { DIGIT } },
{ "!.2.3", '!', new CharType[] { DIGIT } },
{ "1.!.3", '!', new CharType[] { DIGIT } },
{ "1.2.!", '!', new CharType[] { DIGIT } },
{ "v1.2.3", 'v', new CharType[] { DIGIT } },
{ "1.2.3-", null, new CharType[] { DIGIT, LETTER, HYPHEN } },
{ "1.2. 3", ' ', new CharType[] { DIGIT } },
{ "1.2.3=alpha", '=', new CharType[] { HYPHEN, PLUS, EOL } },
{ "1.2.3~beta", '~', new CharType[] { HYPHEN, PLUS, EOL } },
{ "1.2.3-be$ta", '$', new CharType[] { PLUS, EOL } },
{ "1.2.3+b1+b2", '+', new CharType[] { EOL } },
{ "1.2.3-rc!", '!', new CharType[] { PLUS, EOL } },
{ "1.2.3-+", '+', new CharType[] { DIGIT, LETTER, HYPHEN } },
{ "1.2.3-@", '@', new CharType[] { DIGIT, LETTER, HYPHEN } },
{ "1.2.3+@", '@', new CharType[] { DIGIT, LETTER, HYPHEN } },
{ "1.2.3-rc1.", null, new CharType[] { DIGIT, LETTER, HYPHEN } },
{ "1.2.3+20140620.", null, new CharType[] { DIGIT, LETTER, HYPHEN } },
{ "1.2.3-b.+b", '+', new CharType[] { DIGIT, LETTER, HYPHEN } },
{ "1.2.3-rc..", '.', new CharType[] { DIGIT, LETTER, HYPHEN } },
{ "1.2.3-rc+bld..", '.', new CharType[] { DIGIT, LETTER, HYPHEN } },
});
}
}