751 lines
26 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.concurrent.Callable;
import java.util.function.Consumer;
import sun.security.util.ManifestDigester;
import org.testng.annotations.Test;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Factory;
import static java.nio.charset.StandardCharsets.UTF_8;
import static org.testng.Assert.*;
/**
* @test
* @bug 8217375
* @modules java.base/sun.security.util:+open
* @compile ../../tools/jarsigner/Utils.java
* @run testng/othervm FindSection
* @summary Check {@link ManifestDigester#findSection}.
*/
public class FindSection {
/*
* TODO:
* FIXED_8217375 is not intended to keep. it is intended to show what
* exactly has changed with respect to the previous version for which no
* such test existed.
*/
static final boolean FIXED_8217375 = true;
/**
* {@link ManifestDigester.Entry#digestWorkaround} should not feed the
* trailing blank line into the digester. Before resolution of 8217375 it
* fed the trailing blank line into the digest if the second line break
* was at the end of the file due to <pre>
* if (allBlank || (i == len-1)) {
* if (i == len-1)
* pos.endOfSection = i;
* else
* pos.endOfSection = last;
* </pre> in {@link ManifestDigester#findSection}. In that case at the end
* of the manifest file, {@link ManifestDigester.Entry#digestWorkaround}
* would have produced the same digest as
* {@link ManifestDigester.Entry#digest} which was wrong and without effect
* at best.
* <p>
* Once this fix is accepted, this flag can be removed along with
* {@link #actualEndOfSection8217375}.
*/
static final boolean FIXED_8217375_EOF_ENDOFSECTION = FIXED_8217375;
/**
* {@link ManifestDigester.Position.endOfSection} usually points to the
* start position of the blank line trailing a section minus one.
* If a {@link ManifestDigester.Position} returned by
* {@link ManifestDigester#findSection} is based on a portion that starts
* with a blank line, above statement is (or was) not true, because of the
* initialization of {@code last} in {@link ManifestDigester#findSection}
* <pre>
* int last = offset;
* </pre>
* which would point after incrementing it in {@code pos.endOfSection + 1}
* on line 128 (line number before this change) or {@code int sectionLen =
* pos.endOfSection-start+1;} on line 133 (line number before this change)
* at one byte after the first line break character of usually two and
* possibly (assuming "{@code \r\n}" default line break normally) in between
* the two characters of a line break. After subtracting again the index of
* the section start position on former line 133, the last byte would be
* missed to be digested by {@link ManifestDigester.Entry#digestWorkaround}.
* <p>
* All this, however could possibly matter (or have mattered) only when
* {@link ManifestDigester#findSection} was invoked with an offset position
* pointing straight to a line break which happens if a manifest starts
* with an empty line or if there are superfluous blank lines between
* sections in both cases no useful manifest portion is identified.
* Superfluous blank lines are not identified as sections (because they
* don't have a name and specifically don't meet {@code if (len > 6) {} on
* former line 136. Manifests starting with a line break are not any more
* useful either.
* <p>
* Once this fix is accepted, this flag can be removed along with
* {@link #actualEndOfSection8217375}.
*/
static final boolean FIXED_8217375_STARTWITHBLANKLINE_ENDOFSECTION =
FIXED_8217375;
static Constructor<?> PositionConstructor;
static Method findSection;
static Field rawBytes;
static Field endOfFirstLine;
static Field endOfSection;
static Field startOfNext;
@BeforeClass
public static void setFindSectionAccessible() throws Exception {
Class<?> Position = Arrays.stream(ManifestDigester.class.
getDeclaredClasses()).filter(c -> c.getSimpleName().
equals("Position")).findFirst().get();
PositionConstructor = Position.getDeclaredConstructor();
PositionConstructor.setAccessible(true);
findSection = ManifestDigester.class.getDeclaredMethod("findSection",
int.class, Position);
findSection.setAccessible(true);
rawBytes = ManifestDigester.class.getDeclaredField("rawBytes");
rawBytes.setAccessible(true);
endOfFirstLine = Position.getDeclaredField("endOfFirstLine");
endOfFirstLine.setAccessible(true);
endOfSection = Position.getDeclaredField("endOfSection");
endOfSection.setAccessible(true);
startOfNext = Position.getDeclaredField("startOfNext");
startOfNext.setAccessible(true);
}
static class Position {
final int endOfFirstLine; // not including newline character
final int endOfSection; // end of section, not including the blank line
// between sections
final int startOfNext; // the start of the next section
Position(Object pos) throws ReflectiveOperationException {
endOfFirstLine = FindSection.endOfFirstLine.getInt(pos);
endOfSection = FindSection.endOfSection.getInt(pos);
startOfNext = FindSection.startOfNext.getInt(pos);
}
}
Position findSection(byte[] manifestBytes)
throws ReflectiveOperationException {
ManifestDigester manDig = new ManifestDigester("\n\n".getBytes(UTF_8));
FindSection.rawBytes.set(manDig, manifestBytes);
Object pos = PositionConstructor.newInstance();
Object result = findSection.invoke(manDig, offset, pos);
if (Boolean.FALSE.equals(result)) {
return null; // indicates findSection having returned false
} else {
return new Position(pos);
}
}
@DataProvider(name = "parameters")
public static Object[][] parameters() {
return new Object[][] { { 0 }, { 42 } };
}
@Factory(dataProvider = "parameters")
public static Object[] createTests(int offset) {
return new Object[]{ new FindSection(offset) };
}
final int offset;
FindSection(int offset) {
this.offset = offset;
}
@BeforeMethod
public void verbose() {
System.out.println("offset = " + offset);
}
Position findSection(String manifestString)
throws ReflectiveOperationException {
byte[] manifestBytes = manifestString.getBytes(UTF_8);
byte[] manifestWithOffset = new byte[manifestBytes.length + offset];
System.arraycopy(manifestBytes, 0, manifestWithOffset, offset,
manifestBytes.length);
return findSection(manifestWithOffset);
}
/**
* Surprising, but the offset actually makes a difference in
* {@link ManifestDigester#findSection} return value.
*/
@SuppressWarnings("unused")
int actualEndOfFirstLine8217375(int correctPosition) {
// if the parsed portion of the manifest starts with a blank line,
// and offset is 0, "pos.endOfFirstLine = -1;" probably denoting a
// yet uninitialized value coincides with the assignment by
// "pos.endOfFirstLine = i-1;" if i == 0 and
// "if (pos.endOfFirstLine == -1)" after "case '\n':" happens to
// become true even though already assigned.
if (offset == 0 && correctPosition == -1 && !FIXED_8217375) return 0;
return correctPosition;
}
@SuppressWarnings("unused")
int actualEndOfSection8217375(int correctPosition, boolean eof, int lbl) {
// if the parsed portion of the manifest ends with a blank line and
// just before eof, the blank line is included in Position.endOfSection/
// Section.length (the one usually without blank line as well as in
// Position.startOfNext/Section.lengthWithBlankLine) which is used
// in digestWorkaround (independent of the digest without workaround)
if (eof && !FIXED_8217375_EOF_ENDOFSECTION) {
return correctPosition + lbl;
} else if (correctPosition == -1
&& !FIXED_8217375_STARTWITHBLANKLINE_ENDOFSECTION) {
return 0;
} else {
return correctPosition;
}
}
AssertionError collectErrors(AssertionError a, Runnable run) {
try {
run.run();
} catch (AssertionError e) {
if (a == null) a = new AssertionError();
a.addSuppressed(e);
}
return a;
}
void assertPosition(Position pos,
int endOfFirstLine, int endOfSection, int startOfNext) {
AssertionError a = null;
a = collectErrors(a, () -> assertEquals(
pos.endOfFirstLine, endOfFirstLine + offset, "endOfFirstLine"));
a = collectErrors(a, () -> assertEquals(
pos.endOfSection, endOfSection + offset, "endOfSection"));
a = collectErrors(a, () -> assertEquals(
pos.startOfNext, startOfNext + offset, "startOfNext"));
if (a != null) throw a;
}
void catchCrCausesIndexOutOfBoundsException(
Callable<Position> test, Consumer<Position> asserts) {
try {
Position x = test.call();
if (!FIXED_8217375) fail();
asserts.accept(x);
} catch (Exception e) {
if (e instanceof IndexOutOfBoundsException ||
e.getCause() instanceof IndexOutOfBoundsException) {
if (FIXED_8217375) throw new AssertionError(e);
} else {
throw new AssertionError(e);
}
}
}
@Test
public void testEmpty() throws Exception {
assertNull(findSection(""));
}
@Test
public void testOneLineBreakCr() throws Exception {
catchCrCausesIndexOutOfBoundsException(
() -> findSection("\r"),
p -> assertPosition(p,
-1, actualEndOfSection8217375(-1, false, 1), 1)
);
}
@Test
public void testOneLineBreakLf() throws Exception {
assertPosition(findSection("\n"),
-1, actualEndOfSection8217375(-1, false, 1), 1);
}
@Test
public void testOneLineBreakCrLf() throws Exception {
assertPosition(findSection("\r\n"),
actualEndOfFirstLine8217375(-1),
actualEndOfSection8217375(-1, true, 2),
2);
}
@Test
public void testSpaceAndLineBreakCr() throws Exception {
catchCrCausesIndexOutOfBoundsException(
() -> findSection(" \r"),
p -> assertPosition(p, 2, 3, 4)
);
}
@Test
public void testSpaceAndOneLineBreakLf() throws Exception {
assertPosition(findSection(" \n"), 2, 3, 4);
}
@Test
public void testSpaceAndOneLineBreakCrLf() throws Exception {
assertPosition(findSection(" \r\n"), 2, 4, 5);
}
@Test
public void testOneLineBreakCrAndSpace() throws Exception {
assertPosition(findSection("\r "),
-1, actualEndOfSection8217375(-1, false, 1), 1);
}
@Test
public void testOneLineBreakLfAndSpace() throws Exception {
assertPosition(findSection("\n "),
-1, actualEndOfSection8217375(-1, false, 1), 1);
}
@Test
public void testOneLineBreakCrLfAndSpace() throws Exception {
assertPosition(findSection("\r\n "),
actualEndOfFirstLine8217375(-1),
actualEndOfSection8217375(-1, false, 1),
2);
}
@Test
public void testCrEof() throws Exception {
catchCrCausesIndexOutOfBoundsException(
() -> findSection("abc\r"),
p -> assertPosition(p, 2, 3, 4)
);
}
@Test
public void testLfEof() throws Exception {
assertPosition(findSection("abc\n"), 2, 3, 4);
}
@Test
public void testCrLfEof() throws Exception {
assertPosition(findSection("abc\r\n"), 2, 4, 5);
}
@Test
public void testCrContinued() throws Exception {
assertPosition(findSection("abc\rxyz\r\n\r\n "), 2, 8, 11);
}
@Test
public void testLfContinued() throws Exception {
assertPosition(findSection("abc\nxyz\r\n\r\n "), 2, 8, 11);
}
@Test
public void testCrLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n "), 2, 9, 12);
}
@Test
public void testCrCrEof() throws Exception {
catchCrCausesIndexOutOfBoundsException(
() -> findSection("abc\r\nxyz\r\r"),
p -> assertPosition(p,
2, actualEndOfSection8217375(8, true, 1), 10)
);
}
@Test
public void testCrCrContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\r "), 2, 8, 10);
}
@Test
public void testLfLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\n\n"),
2, actualEndOfSection8217375(8, true, 1), 10);
}
@Test
public void testLfLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\n\n "), 2, 8, 10);
}
@Test
public void testCrLfEof2() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n"), 2, 9, 10);
}
@Test
public void testMainSectionNotTerminatedWithLineBreak() throws Exception {
assertNull(findSection("abc\r\nxyz\r\n "));
}
@Test
public void testLfCrEof() throws Exception {
catchCrCausesIndexOutOfBoundsException(
() -> findSection("abc\r\nxyz\n\r"),
p -> assertPosition(p,
2, actualEndOfSection8217375(8, true, 1), 10)
);
}
@Test
public void testLfCrContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\n\r "), 2, 8, 10);
}
@Test
public void testCrLfCrEof() throws Exception {
catchCrCausesIndexOutOfBoundsException(
() -> findSection("abc\r\nxyz\r\n\r"),
p -> assertPosition(p,
2, actualEndOfSection8217375(9, true, 2), 11)
);
}
@Test
public void testCrLfCrContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r "), 2, 9, 11);
}
@Test
public void testCrLfLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n"),
2, actualEndOfSection8217375(9, true, 1), 11);
}
@Test
public void testCrLfLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n "), 2, 9, 11);
}
@Test
public void testCrLfCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n"),
2, actualEndOfSection8217375(9, true, 2), 12);
}
@Test
public void testCrLfCfLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n "), 2, 9, 12);
}
@Test
public void testCrLfCrCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r"), 2, 9, 11);
}
@Test
public void testCrLfCrCrContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r "), 2, 9, 11);
}
@Test
public void testCrLfLfCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r"), 2, 9, 11);
}
@Test
public void testCrLfLfCrContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r "), 2, 9, 11);
}
@Test
public void testCrLfCrLfCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\r"), 2, 9, 12);
}
@Test
public void testCrLfCfLfCrContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\r "), 2, 9, 12);
}
@Test
public void testCrLfCrLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n "), 2, 9, 12);
}
@Test
public void testCrLfLfLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\n"), 2, 9, 11);
}
@Test
public void testCrLfLfLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\n "), 2, 9, 11);
}
@Test
public void testCrLfCrLfLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\n "), 2, 9, 12);
}
@Test
public void testCrLfCrCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r\n"), 2, 9, 11);
}
@Test
public void testCrLfCrCrLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r\n "), 2, 9, 11);
}
@Test
public void testCrLfLfCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r\n"), 2, 9, 11);
}
@Test
public void testCrLfLfCrLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r\n "), 2, 9, 11);
}
@Test
public void testCrLfCrLfCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\r\n"), 2, 9, 12);
}
@Test
public void testCrLfCfLfCrLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\r\n "), 2, 9, 12);
}
@Test
public void testCrLfLfCrCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r\r"), 2, 9, 11);
}
@Test
public void testCrLfCrLfCrCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\r\r"), 2, 9, 12);
}
@Test
public void testCrLfCrLfCrContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\r "), 2, 9, 12);
}
@Test
public void testCrLfLfLfCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\n\r"), 2, 9, 11);
}
@Test
public void testCrLfLfCrLfCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r\n\r"), 2, 9, 11);
}
@Test
public void testCrLfLfLfLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\n\n"), 2, 9, 11);
}
@Test
public void testCrLfLfCrLfLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r\n\n"), 2, 9, 11);
}
@Test
public void testCrLfLfCrCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r\r\n"), 2, 9, 11);
}
@Test
public void testCrLfCrLfCrCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\r\r\n"), 2, 9, 12);
}
@Test
public void testCrLfCrLfCrLfContinued() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\n\r\n "), 2, 9, 12);
}
@Test
public void testCrLfLfLfCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\n\r\n"), 2, 9, 11);
}
@Test
public void testCrLfLfCrLfCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\n\r\n\r\n"), 2, 9, 11);
}
@Test
public void testCrLfCrCrLfCrCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r\n\r"), 2, 9, 11);
}
@Test
public void testCrLfCrCrCrCrEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r\r"), 2, 9, 11);
}
@Test
public void testCrLfCrCrLfLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r\n\n"), 2, 9, 11);
}
@Test
public void testCrLfCrCrLfCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r\n\r\n"), 2, 9, 11);
}
@Test
public void testCrLfCrCrCrLfEof() throws Exception {
assertPosition(findSection("abc\r\nxyz\r\n\r\r\r\n"), 2, 9, 11);
}
/*
* endOfFirstLine is the same regardless of the line break delimiter
*/
@Test
public void testEndOfFirstLineVsLineBreak() throws Exception {
for (String lb : new String[] { "\r", "\n", "\r\n" }) {
Position p = findSection("abc" + lb + "xyz" + lb + lb + " ");
// main assertion showing endOfFirstLine independent of line break
assertEquals(p.endOfFirstLine, 2 + offset);
// assert remaining positions as well just for completeness
assertPosition(p, 2, 5 + 2 * lb.length(), 6 + 3 * lb.length());
}
}
/*
* '\r' at the end of the bytes causes index out of bounds exception
*/
@Test
public void testCrLastCausesIndexOutOfBounds() throws Exception {
catchCrCausesIndexOutOfBoundsException(
() -> findSection("\r"),
p -> assertPosition(p,
-1, actualEndOfSection8217375(-1, true, 1), 1)
);
}
/*
* endOfSection includes second line break if at end of bytes only
*/
@Test
public void testEndOfSectionWithLineBreakVsEof() throws Exception {
AssertionError errors = new AssertionError("offset = " + offset);
for (String lb : new String[] { "\r", "\n", "\r\n" }) {
for (boolean eof : new boolean[] { false, true }) {
Position p;
try {
p = findSection("abc" + lb + lb + (eof ? "" : "xyz"));
} catch (RuntimeException | ReflectiveOperationException e) {
if ((e instanceof IndexOutOfBoundsException ||
e.getCause() instanceof IndexOutOfBoundsException)
&& eof && "\r".equals(lb) && !FIXED_8217375) continue;
throw e;
}
AssertionError a = new AssertionError("offset = " + offset
+ ", lb = " + Utils.escapeStringWithNumbers(lb) + ", "
+ "eof = " + eof);
// main assertion showing endOfSection including second line
// break when at end of file
a = collectErrors(a, () -> assertEquals(
p.endOfSection,
actualEndOfSection8217375(
2 + lb.length() + offset, eof, lb.length()) ));
// assert remaining positions as well just for completeness
a = collectErrors(a, () -> assertPosition(p,
2,
actualEndOfSection8217375(
2 + lb.length(), eof, lb.length()),
3 + lb.length() * 2));
if (a.getSuppressed().length > 0) errors.addSuppressed(a);
}
}
if (errors.getSuppressed().length > 0) throw errors;
}
/*
* returns position even if only one line break before end of bytes.
* because no name will be found the result will be skipped and no entry
* will be created.
*/
@Test
public void testReturnPosVsEof() throws Exception {
for (String lb : new String[] { "\r", "\n", "\r\n" }) {
for (boolean eof : new boolean[] { false, true }) {
try {
Position p = findSection("abc" + lb + (eof ? "" : "xyz"));
assertTrue(p != null == eof);
} catch (RuntimeException | ReflectiveOperationException e) {
if ((e instanceof IndexOutOfBoundsException ||
e.getCause() instanceof IndexOutOfBoundsException)
&& eof && "\r".equals(lb) && !FIXED_8217375) continue;
throw e;
}
}
}
}
/*
* it could be normally be expected that startOfNext would point to the
* start of the next section after a blank line but that is not the case
* if a section ends with only one line break and no blank line immediately
* before eof of the manifest.
* such an entry will be digested without the trailing blank line which is
* only fine until another section should be added afterwards.
*/
@Test
public void testStartOfNextPointsToEofWithNoBlankLine() throws Exception {
for (String lb : new String[] { "\r", "\n", "\r\n" }) {
for (boolean blank : new boolean[] { false, true }) {
String manifest = "abc" + lb + "xyz" + lb + (blank ? lb : "");
try {
Position p = findSection(manifest);
// assert that startOfNext points to eof in all cases
// whether with or without a blank line before eof
assertEquals(p.startOfNext, manifest.length() + offset);
// assert remaining positions as well just for completeness
assertPosition(p,
2,
actualEndOfSection8217375(
5 + lb.length() * 2,
true,
blank ? lb.length() : 0),
manifest.length());
} catch (RuntimeException | ReflectiveOperationException e) {
if ((e instanceof IndexOutOfBoundsException ||
e.getCause() instanceof IndexOutOfBoundsException)
&& "\r".equals(lb) && !FIXED_8217375) continue;
throw e;
}
}
}
}
}