From c1597839ba862a0e27eb2b43307a55841854284c Mon Sep 17 00:00:00 2001 From: bow <bow@bow.web.id> Date: Wed, 8 Apr 2015 14:43:07 +0200 Subject: [PATCH] Add utility function for parsing numbers --- .../nl/lumc/sasc/biopet/utils/package.scala | 59 +++++++++++++++++++ 1 file changed, 59 insertions(+) create mode 100644 public/biopet-framework/src/main/scala/nl/lumc/sasc/biopet/utils/package.scala diff --git a/public/biopet-framework/src/main/scala/nl/lumc/sasc/biopet/utils/package.scala b/public/biopet-framework/src/main/scala/nl/lumc/sasc/biopet/utils/package.scala new file mode 100644 index 000000000..eab8c3764 --- /dev/null +++ b/public/biopet-framework/src/main/scala/nl/lumc/sasc/biopet/utils/package.scala @@ -0,0 +1,59 @@ +package nl.lumc.sasc.biopet + +import scala.util.{ Failure, Success, Try } + +/** + * General utility functions. + * + * @author Wibowo Arindrarto <w.arindrarto@lumc.nl> + */ +package object utils { + + /** Regular expression for matching entire integer numbers (numbers without decimals / fractions) */ + val isInteger = """^([-+]?\d+)L?$""".r + + /** Regular expression for matching entire decimal numbers (compatible with the scientific notation) */ + val isDecimal = """^([-+]?\d*\.?\d+(?:[eE][-+]?[0-9]+)?)$""".r + + /** + * Tries to convert the given string with the given conversion functions recursively. + * + * If conversion is successful, the converted object is returned within as a [[Success]] type. Otherwise, a [[Failure]] + * is returned. The order of conversion functions is the same as the order they are specified. + * + * @param raw the string to convert. + * @param funcs one or more conversion functions to apply. + * @return a [[Try]] object encapsulating the conversion result. + */ + def tryToConvert(raw: String, funcs: (String => Any)*): Try[Any] = funcs match { + + case Seq(firstFunc, otherFuncs @ _*) => + Try(firstFunc(raw)) + .transform(s => Success(s), f => tryToConvert(raw, otherFuncs: _*)) + + case Nil => Try(throw new Exception(s"Can not extract value from string $raw")) + } + + /** + * Tries to convert the given string into the appropriate number representation. + * + * The given string must be whole numbers without any preceeding or succeeding whitespace. This function takes + * into account the maximum values of the number object to use. For example, if the raw string represents a bigger + * number than the maximum [[Int]] value, then a [[Long]] will be used. If the number is still bigger than a [[Long]], + * the [[BigInt]] class will be used. The same is applied for decimal numbers, where the conversion order is first + * a [[Double]], then a [[BigDecimal]]. + * + * @param raw the string to convert. + * @return a [[Try]] object encapsulating the conversion result. + */ + def tryToParseNumber(raw: String) = raw match { + + case isInteger(i) => + tryToConvert(i, x => x.toInt, x => x.toLong, x => BigInt(x)) + + case isDecimal(f) => + tryToConvert(f, x => x.toDouble, x => BigDecimal(x)) + + case otherwise => Try(throw new Exception(s"Can not extract number from string $raw")) + } +} -- GitLab