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

sbt.CommandUtil.scala Maven / Gradle / Ivy

The newest version!
/*
 * sbt
 * Copyright 2023, Scala center
 * Copyright 2011 - 2022, Lightbend, Inc.
 * Copyright 2008 - 2010, Mark Harrah
 * Licensed under Apache License 2.0 (see LICENSE)
 */

package sbt

import java.io.File
import java.util.regex.{ Pattern, PatternSyntaxException }

import sbt.internal.util.AttributeKey
import sbt.internal.util.complete.Parser
import sbt.internal.util.complete.DefaultParsers._
import sbt.internal.util.Util.nilSeq

import sbt.io.IO
import sbt.io.syntax._

object CommandUtil {
  def readLines(files: Seq[File]): Seq[String] =
    files flatMap (IO.readLines(_)) flatMap processLine
  def processLine(s: String): Option[String] = {
    val s2 = s.trim; if (ignoreLine(s2)) None else Some(s2)
  }
  def ignoreLine(s: String): Boolean = s.isEmpty || s.startsWith("#")

  private def canRead = (_: File).canRead
  def notReadable(files: Seq[File]): Seq[File] = files filterNot canRead
  def readable(files: Seq[File]): Seq[File] = files filter canRead

  // slightly better fallback in case of older launcher
  def bootDirectory(state: State): File =
    try state.configuration.provider.scalaProvider.launcher.bootDirectory
    catch { case _: NoSuchMethodError => new File(".").getAbsoluteFile }

  def aligned(pre: String, sep: String, in: Seq[(String, String)]): Seq[String] =
    if (in.isEmpty) nilSeq
    else {
      val width = in.iterator.map(_._1.length).max
      for ((a, b) <- in) yield pre + fill(a, width) + sep + b
    }

  def fill(s: String, size: Int): String = s + " " * math.max(size - s.length, 0)

  def withAttribute[T](s: State, key: AttributeKey[T], ifMissing: String)(f: T => State): State =
    s get key match {
      case None =>
        s.log.error(ifMissing); s.fail
      case Some(nav) => f(nav)
    }

  def singleArgument(exampleStrings: Set[String]): Parser[String] = {
    val arg = (NotSpaceClass ~ any.*) map { case (ns, s) => (ns +: s).mkString }
    token(Space) ~> token(arg examples exampleStrings)
  }

  def detail(selected: String, detailMap: Map[String, String]): String =
    detailMap.get(selected) match {
      case Some(exactDetail) => exactDetail
      case None =>
        try {
          val details = searchHelp(selected, detailMap)
          if (details.isEmpty)
            "No matches for regular expression '" + selected + "'."
          else
            layoutDetails(details)
        } catch {
          case pse: PatternSyntaxException =>
            sys.error("Invalid regular expression (java.util.regex syntax).\n" + pse.getMessage)
        }
    }

  def searchHelp(selected: String, detailMap: Map[String, String]): Map[String, String] = {
    val pattern = Pattern.compile(selected, HelpPatternFlags)
    detailMap flatMap {
      case (k, v) =>
        val contentMatches = Highlight.showMatches(pattern)(v)
        val keyMatches = Highlight.showMatches(pattern)(k)
        val keyString = Highlight.bold(keyMatches getOrElse k)
        val contentString = contentMatches getOrElse v
        if (keyMatches.isDefined || contentMatches.isDefined)
          Seq((keyString, contentString))
        else
          nilSeq
    }
  }

  def layoutDetails(details: Map[String, String]): String =
    details.map { case (k, v) => k + "\n\n  " + v }.mkString("\n", "\n\n", "\n")

  final val HelpPatternFlags = Pattern.CASE_INSENSITIVE | Pattern.UNICODE_CASE

  private[sbt] def isSbtBuild(baseDir: File) =
    (baseDir / "project").exists() || (baseDir * "*.sbt").get().nonEmpty
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy