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

com.ossuminc.riddl.command.PassCommand.scala Maven / Gradle / Ivy

/*
 * Copyright 2019 Ossum, Inc.
 *
 * SPDX-License-Identifier: Apache-2.0
 */

package com.ossuminc.riddl.command

import com.ossuminc.riddl.language.Messages
import com.ossuminc.riddl.language.Messages.Messages
import com.ossuminc.riddl.language.parsing.TopLevelParser
import com.ossuminc.riddl.language.parsing.RiddlParserInput
import com.ossuminc.riddl.utils.{CommonOptions, Logger, PathUtils, PlatformContext, URL, Await}
import com.ossuminc.riddl.passes.{Pass, PassCreators, PassInput, PassesResult}

import java.nio.file.Path
import scala.reflect.ClassTag
import scala.concurrent.{ExecutionContext, Future}
import scala.concurrent.duration.DurationInt

trait PassCommandOptions extends CommandOptions {
  def outputDir: Option[Path]
  override def check: Messages = {
    val msgs1 = super.check
    val msgs2 = if outputDir.isEmpty then {
      Messages.errors("An output directory was not provided.")
    } else {
      Messages.empty
    }
    msgs1 ++ msgs2
  }
}

/** An abstract base class for commands that use passes.
  *
  * @param name
  *   The name of the command to pass to [[Command]]
  * @tparam OPT
  *   The option type for the command
  */
abstract class PassCommand[OPT <: PassCommandOptions: ClassTag](name: String)(using pc: PlatformContext)
    extends Command[OPT](name) {

  /** Get the passes to run given basic input for pass creation
    *
    * @param log
    *   The log to use
    * @param commonOptions
    *   The common options for the command
    * @param options
    *   The command options
    * @return
    *   A [[com.ossuminc.riddl.passes.PassCreator]] function that creates the pass
    */
  def getPasses(
    options: OPT
  ): PassCreators

  /** A method to override the options */
  def overrideOptions(options: OPT, newOutputDir: Path): OPT

  private final def doRun(
    options: OPT
  ): Either[Messages, PassesResult] = {
    options.withInputFile { (inputPath: Path) =>
      val url = PathUtils.urlFromPath(inputPath)
      implicit val ec: ExecutionContext = pc.ec
      val future = RiddlParserInput.fromURL(url).map { rpi =>
        TopLevelParser.parseInput(rpi) match {
          case Left(errors) =>
            Left[Messages, PassesResult](errors)
          case Right(root) =>
            val input: PassInput = PassInput(root)
            val passes = getPasses(options)
            val result = Pass.runThesePasses(input, passes)
            if result.messages.hasErrors then Left(result.messages)
            else
              if pc.options.debug then
                println(s"Errors after running ${this.name}:")
                println(result.messages.format)
              Right(result)
        }
      }
      Await.result(future, 20.seconds)
    }
  }

  /** The basic implementation of the command. This should be called with `super.run(...)` from the subclass
    * implementation.
    * @param originalOptions
    *   The original options to the command
    * @param commonOptions
    *   The options common to all commands
    * @param log
    *   A log for logging errors, warnings, and info
    * @param outputDirOverride
    *   Any override to the outputDir option from the command line
    * @return
    *   Either a set of Messages on error or a Unit on success
    */
  override def run(
    originalOptions: OPT,
    outputDirOverride: Option[Path]
  ): Either[Messages, PassesResult] = {
    val options = if outputDirOverride.nonEmpty then
      val path = outputDirOverride.fold(Path.of(""))(identity)
      overrideOptions(originalOptions, path)
    else originalOptions

    val messages = options.check

    if messages.nonEmpty then {
      Left[Messages, PassesResult](messages) // no point even parsing if there are option errors
    } else {
      doRun(options)
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy