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

scalaParser.subscript.ast.Operators.scala Maven / Gradle / Ivy

package scalaParser.subscript.ast


trait Operators {this: Ast =>
  import Constants.DSL.Op._
  import Constants._

  trait ScriptOperator extends Node {
    val method: String
  }

  trait ScriptOperatorNary extends ScriptOperator {
    val operands: Seq[Node]

    override def rewrite(implicit context: Context, output: Output): String =
      if (operands.size > 1) {
        def operandsStr = operands.map(_.compile).mkString(", ")
        s"$method($operandsStr)"
      }
      // If there's just one operand, compile it without wrapping in a DSL method
      // If there are no operands, return an empty string
      else operands.headOption.map(_.compile).getOrElse("")
  }

  case class IdentityOp(node: Node) extends Expr7 with IdentityNode {
    val operands = Nil
    val method   = ""
  }


  trait ExprSeq extends ScriptOperatorNary {val method = SEQUENCE}

  trait Expr9 extends Node
  case class Expr9Seq     (operands: Seq[Node]) extends Expr9 with ExprSeq
  case class Expr9Identity(node    :     Node ) extends Expr9 with IdentityNode

  case class Expr8(operands: Seq[Node]) extends ExprSeq

  trait Expr7 extends ScriptOperator

  trait IfExprBase extends Expr7 {
    val nIf: Node

    def condition: String = 
      s"""$method (${Name.NODE} => {
         |  implicit val ${Name.HERE} = ${Name.NODE}
         |${nIf.compile}
         |})""".stripMargin
  }

  case class IfExpr(nIf: Node, nThen: Node) extends IfExprBase {
    val method = IF_EXPR

    def rewrite(implicit context: Context, output: Output): String =
      s"$condition(${nThen.compile})"
  }

  case class IfElseExpr(nIf: Node, nThen: Node, nElse: Node) extends IfExprBase {
    val method = IF_ELSE_EXPR

    def rewrite(implicit context: Context, output: Output): String =
      s"$condition(${nThen.compile}, ${nElse.compile})"
  }

  case class DoThen    (operands: Seq[Node]) extends Expr7 with ScriptOperatorNary {val method = DO_THEN     }
  case class DoThenElse(operands: Seq[Node]) extends Expr7 with ScriptOperatorNary {val method = DO_THEN_ELSE}
  case class DoElse    (operands: Seq[Node]) extends Expr7 with ScriptOperatorNary {val method = DO_ELSE     }

  trait Expr6 extends ScriptOperatorNary
  case class OrPar1(operands: Seq[Node]) extends Expr6 {val method = OR_PAR_1}
  case class OrPar2(operands: Seq[Node]) extends Expr6 {val method = OR_PAR_2}

  trait Expr5 extends ScriptOperatorNary
  case class AndPar1(operands: Seq[Node]) extends Expr5 {val method = AND_PAR_1}
  case class AndPar2(operands: Seq[Node]) extends Expr5 {val method = AND_PAR_2}

  trait Expr4 extends ScriptOperatorNary
  case class Equality(operands: Seq[Node]) extends Expr4 {val method = EQUALITY}

  trait Expr3 extends ScriptOperatorNary
  case class Alternative(operands: Seq[Node]) extends Expr3 {val method = ALTERNATIVE}

  trait Expr2 extends ScriptOperatorNary
  case class Disrupt         (operands: Seq[Node]) extends Expr2 {val method = DISRUPT            }
  case class Shuffle         (operands: Seq[Node]) extends Expr2 {val method = SHUFFLE            }
  case class Shuffle1OrMore  (operands: Seq[Node]) extends Expr2 {val method = SHUFFLE_1_OR_MORE  }
  case class Interrupt       (operands: Seq[Node]) extends Expr2 {val method = INTERRUPT          }
  case class Interrupt0OrMore(operands: Seq[Node]) extends Expr2 {val method = INTERRUPT_0_OR_MORE}

  case class Expr1(operands: Seq[Node]) extends ExprSeq


  case class Dataflow(nDo: Node, nThen: Seq[DataflowClause], nElse: Seq[DataflowClause]) extends ScriptOperator {
    val method = DATAFLOW

    def rewrite(implicit context: Context, output: Output): String = {
      val nDoStr       = nodeToScript("~~>", nDo)
      val nThenClauses = nThen.map(_.compile).mkString("\n")
      val nElseClauses = nElse.map(_.compile).mkString("\n")

      val defaultMatcher = """case _ => throw new RuntimeException("No suitable matcher found")"""
      def block(content: String) =
        s"""{
           |$content
           |${if (content.isEmpty) defaultMatcher else ""}
           |}""".stripMargin

      if (nThen.isEmpty && nElse.isEmpty) nDo.compile
      else ScriptCall(Literal(
        s"""$method(
           |  $nDoStr
           |, ${partialFunction(nThenClauses)}
           |, ${partialFunction(nElseClauses)}
           |)""".stripMargin
      )).compile
    } 
  }

  trait AbstractPatmatchClause extends Node {
    def lhs(implicit context: Context, output: Output): String
    def rhs(implicit context: Context, output: Output): String

    def rewrite(implicit context: Context, output: Output): String =
      s"case $lhs => $rhs"
  }

  case class DataflowClause(pattern: String, expr: Node, thenClause: Boolean = true) extends AbstractPatmatchClause {
    def lhs(implicit context: Context, output: Output): String = pattern    
    def rhs(implicit context: Context, output: Output): String = nodeToScript(Name.LAMBDA, expr)
  }


  trait AbstractDataflowMap extends Term {
    val method = DATAFLOW_MAP    
  }

  case class DataflowMap(nDo: Node, nThen: Seq[DataflowMapClause]) extends AbstractDataflowMap {
    def rewrite(implicit context: Context, output: Output): String = {
      val nDoStr       = nodeToScript("~~^", nDo)
      val nThenClauses = nThen.map(_.compile).mkString("\n")

      ScriptCall(Literal(
        s"""$method(
           |  $nDoStr
           |, (${partialFunction(nThenClauses)}: Any => Any)
           |)""".stripMargin
      )).compile
    }

  }

  case class DataflowMapShort(nDo: Node, nThen: String) extends AbstractDataflowMap {
    def rewrite(implicit context: Context, output: Output): String = {
      val nDoStr   = nodeToScript("~~^", nDo)

      ScriptCall(Literal(
        s"""$method($nDoStr, $nThen)"""
      )).compile
    }
  }

  case class DataflowMapClause(pattern: String, expr: String) extends AbstractPatmatchClause {
    def lhs(implicit context: Context, output: Output): String = pattern    
    def rhs(implicit context: Context, output: Output): String = expr
  }

  trait Term extends Node
  case class Parenthesised(node: Node) extends Term with IdentityNode
  case class Launch       (node: Node) extends Term with  WrappedNode {val method = LAUNCH       }
  case class LaunchAnchor (node: Node) extends Term with  WrappedNode {val method = LAUNCH_ANCHOR}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy