Skip to content

Commit db03589

Browse files
authored
Merge pull request #209 from ashawley/issue-4520
Fix OOM for ConstructingParser
2 parents e6c490e + e696ee6 commit db03589

File tree

3 files changed

+68
-7
lines changed

3 files changed

+68
-7
lines changed

jvm/src/test/scala/scala/xml/XMLTest.scala

+62-1
Original file line numberDiff line numberDiff line change
@@ -633,7 +633,7 @@ class XMLTestJVM {
633633
import scala.xml.parsing._
634634
@UnitTest
635635
def dontLoop: Unit = {
636-
val xml = "<!DOCTYPE xmeml SYSTEM> <xmeml> <sequence> </sequence> </xmeml> "
636+
val xml = "<!DOCTYPE xmeml SYSTEM 'uri'> <xmeml> <sequence> </sequence> </xmeml> "
637637
val sink = new PrintStream(new ByteArrayOutputStream())
638638
(Console withOut sink) {
639639
(Console withErr sink) {
@@ -765,4 +765,65 @@ class XMLTestJVM {
765765
val formatted = pp.format(x)
766766
assertEquals(x, XML.loadString(formatted))
767767
}
768+
769+
def toSource(s: String) = new scala.io.Source {
770+
val iter = s.iterator
771+
override def reportError(pos: Int, msg: String, out: java.io.PrintStream = Console.err): Unit = {}
772+
}
773+
774+
@UnitTest
775+
def xTokenTest {
776+
val x = xml.parsing.ConstructingParser.fromSource(toSource("a"), false)
777+
assertEquals((): Unit, x.xToken('b'))
778+
}
779+
780+
@UnitTest(expected = classOf[FatalError])
781+
def xCharDataFailure {
782+
val x = xml.parsing.ConstructingParser.fromSource(toSource(""), false)
783+
784+
x.xCharData
785+
}
786+
787+
@UnitTest(expected = classOf[FatalError])
788+
def xCommentFailure {
789+
val x = xml.parsing.ConstructingParser.fromSource(toSource(""), false)
790+
791+
x.xComment
792+
}
793+
794+
@UnitTest
795+
def xmlProcInstrTest {
796+
val x = xml.parsing.ConstructingParser.fromSource(toSource("aa"), false)
797+
798+
assertEquals(new UnprefixedAttribute("aa", Text(""), Null), x.xmlProcInstr)
799+
}
800+
801+
@UnitTest(expected = classOf[FatalError])
802+
def notationDeclFailure {
803+
val x = xml.parsing.ConstructingParser.fromSource(toSource(""), false)
804+
805+
x.notationDecl
806+
}
807+
808+
@UnitTest
809+
def pubidLiteralTest {
810+
val x = xml.parsing.ConstructingParser.fromSource(toSource(""), false)
811+
812+
assertEquals("", x.pubidLiteral)
813+
}
814+
815+
@UnitTest
816+
def xAttributeValueTest {
817+
val x = xml.parsing.ConstructingParser.fromSource(toSource("'"), false)
818+
819+
assertEquals("", x.xAttributeValue)
820+
}
821+
822+
@UnitTest
823+
def xEntityValueTest {
824+
val x = xml.parsing.ConstructingParser.fromSource(toSource(""), false)
825+
826+
assertEquals("", x.xEntityValue)
827+
}
828+
768829
}

shared/src/main/scala/scala/xml/parsing/MarkupParser.scala

+2-2
Original file line numberDiff line numberDiff line change
@@ -397,7 +397,7 @@ trait MarkupParser extends MarkupParserCommon with TokenTests {
397397
} else sb.append(ch)
398398
nextch()
399399
}
400-
throw truncatedError("broken comment")
400+
truncatedError("broken comment")
401401
}
402402

403403
/* todo: move this into the NodeBuilder class */
@@ -928,7 +928,7 @@ trait MarkupParser extends MarkupParserCommon with TokenTests {
928928
new PublicID(pubID, sysID)
929929
} else {
930930
reportSyntaxError("PUBLIC or SYSTEM expected")
931-
scala.sys.error("died parsing notationdecl")
931+
truncatedError("died parsing notationdecl")
932932
}
933933
xSpaceOpt()
934934
xToken('>')

shared/src/main/scala/scala/xml/parsing/MarkupParserCommon.scala

+4-4
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ import Utility.SU
1919
* All members should be accessed through those.
2020
*/
2121
private[scala] trait MarkupParserCommon extends TokenTests {
22-
protected def unreachable = scala.sys.error("Cannot be reached.")
22+
protected def unreachable = truncatedError("Cannot be reached.")
2323

2424
// type HandleType // MarkupHandler, SymbolicXMLBuilder
2525
type InputType // Source, CharArrayReader
@@ -62,7 +62,7 @@ private[scala] trait MarkupParserCommon extends TokenTests {
6262
val buf = new StringBuilder
6363
while (ch != endCh && !eof) {
6464
// well-formedness constraint
65-
if (ch == '<') return errorAndResult("'<' not allowed in attrib value", "")
65+
if (ch == '<') reportSyntaxError("'<' not allowed in attrib value")
6666
else if (ch == SU) truncatedError("")
6767
else buf append ch_returning_nextch
6868
}
@@ -241,11 +241,11 @@ private[scala] trait MarkupParserCommon extends TokenTests {
241241
val head = until.head
242242
val rest = until.tail
243243

244-
while (true) {
244+
while (!eof) {
245245
if (ch == head && peek(rest))
246246
return handler(positioner(), sb.toString)
247247
else if (ch == SU || eof)
248-
truncatedError("") // throws TruncatedXMLControl in compiler
248+
truncatedError(s"died parsing until $until") // throws TruncatedXMLControl in compiler
249249

250250
sb append ch
251251
nextch()

0 commit comments

Comments
 (0)