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

org.specs2.io.FromSource.scala Maven / Gradle / Ivy

package org.specs2
package io

import io.FileReader._
import io.Paths._
import control.Throwablex._
import control.Exceptions._
import main.SystemProperties
import control.TraceLocation

/**
 * Get source code from the current point of execution
 * 
 * There are constants in this trait designed to grab exactly the proper stacktraces, corresponding to the code in the specification
 *
 * It must be noted that this only work if the code being executed is in a file which has the same name as the class containing the code.
 * 
 * The source dir is assumed to be "src/test/scala/" by default but this can be modified by setting the "specs2.srcTestDir" System property
 *
 */
private[specs2]
trait FromSource {
  
  private[specs2] lazy val srcDir: String = SystemProperties.getOrElse("srcTestDir", "src/test/scala").dirPath

  /**
   * get some source code by:
   *   - fetching the current stacktrace
   *   - finding the location of the example (6th trace by default)
   */
  def getCode(depth: Int = 6): Either[String, String] = getCodeFromTo(depth, depth)

  /**
   * get some source code by:
   *   - fetching the current stacktrace
   *   - finding the location of the example by taking the trace of the first line and the trace of the last line
   *    (at depth 6 and 9 by default)
   *
   * @return Left(error) if the startTrace and endTrace do not cover the same file
   *         Right(source code) if we find some code in the source file
   *
   */
  def getCodeFromTo(start: Int = 6, end: Int = 9, startLineOffset: Int = -1, endLineOffset: Int = -1): Either[String, String] = {
    val stackTrace = new Exception().getStackTrace()
    val (startTrace, endTrace) = (TraceLocation(stackTrace.apply(start)), TraceLocation(stackTrace.apply(end)))

    if (startTrace.fileName != endTrace.fileName)
      Left("No source file found at "+srcDir+startTrace.path)
    else {
      val (startLine, endLine) = (startTrace.lineNumber+startLineOffset, endTrace.lineNumber+endLineOffset)
      getCodeFromToWithLocation(startLine, endLine, startTrace)
    }
  }

  /**
   * get the source code of an example declared with the eg method as a prefix method
   */
  def getExampleFrom(depth: Int, lineOffset: Int): Either[String, String] = {
    val stackTrace = new Exception().getStackTrace()
    getCodeFromMethodCall(TraceLocation(stackTrace.apply(depth)))
  }

  /**
   * get the source code of an example declared with the eg method as a postfix method
   */
  def getExampleTo(depth: Int, lineOffset: Int): Either[String, String] = {
    val stackTrace = new Exception().getStackTrace()
    getCodeToMethodCall(TraceLocation(stackTrace.apply(depth)))
  }

  /**
   * @return the location of the current stacktrace, possibly filtered with a function
   */
  def location(stackFilter: Seq[StackTraceElement] => Seq[StackTraceElement]): TraceLocation = {
    val stackTrace = new Exception().getStackTrace().toSeq
    val filtered = stackFilter(stackTrace)
    TraceLocation(filtered.headOption.getOrElse(stackTrace(0)))
  }

  /**
   * @return lines of code specified by a start line and an end line in a file given by a TraceLocation
   */
  private def getCodeFromToWithLocation(startLine: Int, endLine: Int = 9, location: TraceLocation): Either[String, String] = {
    val path = srcDir+location.path

    if (endLine < startLine) {
      Left[String, String]("No source file found at "+path)
    } else {
      tryOr {
        val content = readLines(path)
        if (startLine >= 0) {
          val code = ((startLine to endLine) map content).mkString("\n")
          Right[String, String](code): Either[String, String]
        } else Left[String, String]("Unvalid start line: "+startLine+" for file: "+path)

      } { e => Left[String, String]("No source file found at "+path) }
    }
  }

  /**
   * @return lines of code after a method call, as specified by a TraceLocation.
   * The end of the method call is given by the first '}' character that is encountered
   */
  private def getCodeFromMethodCall(location: TraceLocation): Either[String, String] = {
    val (path, line) = (srcDir+location.path, location.lineNumber - 1)

    tryOr {
      val content = readLines(path)
      if (line >= 0) {
        val (start, last) = content.drop(line - 1).span(l => !l.contains("}"))
        Right[String, String]((start ++ last.take(1)).mkString("\n")): Either[String, String]
      } else Left[String, String]("Unvalid start line: "+line+" for file: "+path)

    } { e => Left[String, String]("No source file found at "+path) }
  }

  /**
   * @return lines of code before a method call (a postfix method), as specified by a TraceLocation.
   * The beginning of the call is given by the first '{' character that is encountered
   */
  private def getCodeToMethodCall(location: TraceLocation): Either[String, String] = {
    val (path, line) = (srcDir+location.path, location.lineNumber - 1)

    tryOr {
      val content = readLines(path)
      if (line >= 0) {
        val toMethodCall = content.dropRight(content.size - (line + 1))
        val (start, last) = toMethodCall.reverse.span(l => !l.contains("{"))
        val code = last.take(1) ++ start.reverse
        Right[String, String](code.mkString("\n")): Either[String, String]
      } else Left[String, String]("Unvalid start line: "+line+" for file: "+path)

    } { e => Left[String, String]("No source file found at "+path) }
  }
}

private[specs2]
object FromSource extends FromSource




© 2015 - 2025 Weber Informatics LLC | Privacy Policy