diff --git a/zio-cli-testkit/shared/src/main/scala/zio/cli/CliAssertion.scala b/zio-cli-testkit/shared/src/main/scala/zio/cli/CliAssertion.scala index f0776303..1abb66d0 100644 --- a/zio-cli-testkit/shared/src/main/scala/zio/cli/CliAssertion.scala +++ b/zio-cli-testkit/shared/src/main/scala/zio/cli/CliAssertion.scala @@ -87,7 +87,7 @@ object CliAssertion { )(implicit cliConfig: CliConfig): ZIO[R, Throwable, TestResult] = check(pairs) { case CliRepr(params, assertion) => command - .parse(params, cliConfig) + .parse(params, cliConfig, Nil) .map(Right(_)) .catchAll { case e: ValidationError => ZIO.succeed(Left(e)) @@ -110,7 +110,7 @@ object CliAssertion { )(implicit cliConfig: CliConfig): ZIO[R, Throwable, TestResult] = check(pairs) { case CliRepr(params, assertion) => command - .parse(params, cliConfig) + .parse(params, cliConfig, Nil) .map(TestReturn.convert) .map(Right(_)) .catchSome { case e: ValidationError => diff --git a/zio-cli/js/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala b/zio-cli/js/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala new file mode 100644 index 00000000..ceeb0a4b --- /dev/null +++ b/zio-cli/js/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala @@ -0,0 +1,5 @@ +package zio.cli + +private[cli] trait FileOptionsPlatformSpecific { + val default: FileOptions = FileOptions.Noop +} diff --git a/zio-cli/jvm/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala b/zio-cli/jvm/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala new file mode 100644 index 00000000..086087b9 --- /dev/null +++ b/zio-cli/jvm/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala @@ -0,0 +1,5 @@ +package zio.cli + +private[cli] trait FileOptionsPlatformSpecific { + val default: FileOptions = FileOptions.Live +} diff --git a/zio-cli/jvm/src/test/scala/zio/cli/LiveFileOptionsSpec.scala b/zio-cli/jvm/src/test/scala/zio/cli/LiveFileOptionsSpec.scala new file mode 100644 index 00000000..71e10e05 --- /dev/null +++ b/zio-cli/jvm/src/test/scala/zio/cli/LiveFileOptionsSpec.scala @@ -0,0 +1,3 @@ +package zio.cli + +object LiveFileOptionsSpec extends LiveFileOptionsSpecShared diff --git a/zio-cli/native/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala b/zio-cli/native/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala new file mode 100644 index 00000000..086087b9 --- /dev/null +++ b/zio-cli/native/src/main/scala/zio/cli/FileOptionsPlatformSpecific.scala @@ -0,0 +1,5 @@ +package zio.cli + +private[cli] trait FileOptionsPlatformSpecific { + val default: FileOptions = FileOptions.Live +} diff --git a/zio-cli/native/src/test/scala/zio/cli/LiveFileOptionsSpec.scala b/zio-cli/native/src/test/scala/zio/cli/LiveFileOptionsSpec.scala new file mode 100644 index 00000000..71e10e05 --- /dev/null +++ b/zio-cli/native/src/test/scala/zio/cli/LiveFileOptionsSpec.scala @@ -0,0 +1,3 @@ +package zio.cli + +object LiveFileOptionsSpec extends LiveFileOptionsSpecShared diff --git a/zio-cli/shared/src/main/scala/zio/cli/CliApp.scala b/zio-cli/shared/src/main/scala/zio/cli/CliApp.scala index 161838a1..5f95523b 100644 --- a/zio-cli/shared/src/main/scala/zio/cli/CliApp.scala +++ b/zio-cli/shared/src/main/scala/zio/cli/CliApp.scala @@ -17,7 +17,40 @@ import scala.annotation.tailrec */ sealed trait CliApp[-R, +E, +A] { self => - def run(args: List[String]): ZIO[R, CliError[E], Option[A]] + /** + * [[CliApp]] will use [[FileOptions]] to look for [[Options]] in the file system (on JVM/Native). + * + * TODO : Add support for JS + * + * Example: + * - given: + * - base command: `git` + * - current working directory: `/path/cwd/a/b/c` + * - home directory: `/path/home` + * - then: [[FileOptions]] will look for [[Options]] in the following places: + * - provided command line args + * - `/path/cwd/a/b/c/.git` + * - `/path/cwd/a/b/.git` + * - `/path/cwd/a/.git` + * - `/path/cwd/.git` + * - `/path/.git` + * - `/path/home/.git` + */ + def runWithFileArgs(args: List[String]): ZIO[R & FileOptions, CliError[E], Option[A]] + + /** + * Calls [[runWithFileArgs]] with the default [[FileOptions]] for the current platform. + */ + final def run(args: List[String]): ZIO[R, CliError[E], Option[A]] = + runWithFileArgs(args).provideSomeLayer[R](ZLayer.succeed(FileOptions.default)) + + /** + * Calls [[runWithFileArgs]] with the [[FileOptions.Noop]]. + * + * This means no attempt will be made to pull [[Options]] from the file system. + */ + final def runWithoutFileArgs(args: List[String]): ZIO[R, CliError[E], Option[A]] = + runWithFileArgs(args).provideSomeLayer[R](ZLayer.succeed(FileOptions.Noop)) def config(newConfig: CliConfig): CliApp[R, E, A] @@ -66,7 +99,7 @@ object CliApp { private def printDocs(helpDoc: HelpDoc): UIO[Unit] = printLine(helpDoc.toPlaintext(80)).! - def run(args: List[String]): ZIO[R, CliError[E], Option[A]] = { + override def runWithFileArgs(args: List[String]): ZIO[R & FileOptions, CliError[E], Option[A]] = { def executeBuiltIn(builtInOption: BuiltInOption): ZIO[R, CliError[E], Option[A]] = builtInOption match { case ShowHelp(synopsis, helpDoc) => @@ -125,8 +158,13 @@ object CliApp { case Command.Subcommands(parent, _) => prefix(parent) } - self.command - .parse(prefix(self.command) ++ args, self.config) + (self.command.names.headOption match { + case Some(name) => ZIO.serviceWithZIO[FileOptions](_.getOptionsFromFiles(name)) + case None => ZIO.succeed(Nil) + }) + .flatMap((fromFiles: List[FileOptions.OptionsFromFile]) => + self.command.parse(prefix(self.command) ++ args, self.config, fromFiles) + ) .foldZIO( e => printDocs(e.error) *> ZIO.fail(CliError.Parsing(e)), { diff --git a/zio-cli/shared/src/main/scala/zio/cli/Command.scala b/zio-cli/shared/src/main/scala/zio/cli/Command.scala index f1521250..57400ca0 100644 --- a/zio-cli/shared/src/main/scala/zio/cli/Command.scala +++ b/zio-cli/shared/src/main/scala/zio/cli/Command.scala @@ -46,7 +46,11 @@ sealed trait Command[+A] extends Parameter with Named { self => final def orElseEither[B](that: Command[B]): Command[Either[A, B]] = map(Left(_)) | that.map(Right(_)) - def parse(args: List[String], conf: CliConfig): IO[ValidationError, CommandDirective[A]] + def parse( + args: List[String], + conf: CliConfig, + fromFiles: List[FileOptions.OptionsFromFile] = Nil + ): IO[ValidationError, CommandDirective[A]] final def subcommands[B](that: Command[B])(implicit ev: Reducable[A, B]): Command[ev.Out] = Command.Subcommands(self, that).map(ev.fromTuple2(_)) @@ -110,14 +114,15 @@ object Command { def parse( args: List[String], - conf: CliConfig + conf: CliConfig, + fromFiles: List[FileOptions.OptionsFromFile] ): IO[ValidationError, CommandDirective[(OptionsType, ArgsType)]] = { def parseBuiltInArgs(args: List[String]): IO[ValidationError, CommandDirective[Nothing]] = if (args.headOption.exists(conf.normalizeCase(_) == conf.normalizeCase(self.name))) { val options = BuiltInOption .builtInOptions(self, self.synopsis, self.helpDoc) Options - .validate(options, args.tail, conf) + .validate(options, args.tail, conf, Nil) .map(_._3) .someOrFail( ValidationError( @@ -158,7 +163,7 @@ object Command { } tuple1 = splitForcedArgs(commandOptionsAndArgs) (optionsAndArgs, forcedCommandArgs) = tuple1 - tuple2 <- Options.validate(options, optionsAndArgs, conf) + tuple2 <- Options.validate(options, optionsAndArgs, conf, fromFiles) (optionsError, commandArgs, optionsType) = tuple2 tuple <- self.args.validate(commandArgs ++ forcedCommandArgs, conf).mapError(optionsError.getOrElse(_)) (argsLeftover, argsType) = tuple @@ -202,9 +207,10 @@ object Command { def parse( args: List[String], - conf: CliConfig + conf: CliConfig, + fromFiles: List[FileOptions.OptionsFromFile] ): IO[ValidationError, CommandDirective[B]] = - command.parse(args, conf).map(_.map(f)) + command.parse(args, conf, fromFiles).map(_.map(f)) lazy val synopsis: UsageSynopsis = command.synopsis @@ -222,9 +228,12 @@ object Command { def parse( args: List[String], - conf: CliConfig + conf: CliConfig, + fromFiles: List[FileOptions.OptionsFromFile] ): IO[ValidationError, CommandDirective[A]] = - left.parse(args, conf).catchSome { case ValidationError(CommandMismatch, _) => right.parse(args, conf) } + left.parse(args, conf, fromFiles).catchSome { case ValidationError(CommandMismatch, _) => + right.parse(args, conf, fromFiles) + } lazy val synopsis: UsageSynopsis = UsageSynopsis.Mixed @@ -286,7 +295,8 @@ object Command { def parse( args: List[String], - conf: CliConfig + conf: CliConfig, + fromFiles: List[FileOptions.OptionsFromFile] ): IO[ValidationError, CommandDirective[(A, B)]] = { val helpDirectiveForChild = { val safeTail = args match { @@ -294,7 +304,7 @@ object Command { case _ :: tail => tail } child - .parse(safeTail, conf) + .parse(safeTail, conf, fromFiles) .collect(ValidationError(ValidationErrorType.InvalidArgument, HelpDoc.empty)) { case CommandDirective.BuiltIn(BuiltInOption.ShowHelp(synopsis, helpDoc)) => val parentName = names.headOption.getOrElse("") @@ -316,7 +326,7 @@ object Command { case _ :: tail => tail } child - .parse(safeTail, conf) + .parse(safeTail, conf, fromFiles) .collect(ValidationError(ValidationErrorType.InvalidArgument, HelpDoc.empty)) { case directive @ CommandDirective.BuiltIn(BuiltInOption.ShowWizard(_)) => directive } @@ -326,7 +336,7 @@ object Command { ZIO.succeed(CommandDirective.builtIn(BuiltInOption.ShowWizard(self))) parent - .parse(args, conf) + .parse(args, conf, fromFiles) .flatMap { case CommandDirective.BuiltIn(BuiltInOption.ShowHelp(_, _)) => helpDirectiveForChild orElse helpDirectiveForParent @@ -335,7 +345,7 @@ object Command { case builtIn @ CommandDirective.BuiltIn(_) => ZIO.succeed(builtIn) case CommandDirective.UserDefined(leftover, a) if leftover.nonEmpty => child - .parse(leftover, conf) + .parse(leftover, conf, fromFiles) .mapBoth( { case ValidationError(CommandMismatch, _) => diff --git a/zio-cli/shared/src/main/scala/zio/cli/FileOptions.scala b/zio-cli/shared/src/main/scala/zio/cli/FileOptions.scala new file mode 100644 index 00000000..8b33899d --- /dev/null +++ b/zio-cli/shared/src/main/scala/zio/cli/FileOptions.scala @@ -0,0 +1,62 @@ +package zio.cli + +import zio._ +import java.nio.file.Path + +trait FileOptions { + def getOptionsFromFiles(command: String): UIO[List[FileOptions.OptionsFromFile]] +} +object FileOptions extends FileOptionsPlatformSpecific { + + final case class OptionsFromFile(path: String, rawArgs: List[String]) + + case object Noop extends FileOptions { + override def getOptionsFromFiles(command: String): UIO[List[OptionsFromFile]] = ZIO.succeed(Nil) + } + + case object Live extends FileOptions { + + private def optReadPath(path: Path): UIO[Option[FileOptions.OptionsFromFile]] = + (for { + _ <- ZIO.logDebug(s"Searching for file options in '$path'") + exists <- ZIO.attempt(path.toFile.exists()) + pathString = path.toString + optContents <- + ZIO + .readFile(pathString) + .map(c => FileOptions.OptionsFromFile(pathString, c.split('\n').map(_.trim).filter(_.nonEmpty).toList)) + .when(exists) + } yield optContents) + .catchAllCause(ZIO.logErrorCause(s"Error reading options from file '$path', skipping...", _).as(None)) + + private def getPathAndParents(path: Path): Task[List[Path]] = + for { + parentPath <- ZIO.attempt(Option(path.getParent)) + parents <- parentPath match { + case Some(parentPath) => getPathAndParents(parentPath) + case None => ZIO.succeed(Nil) + } + } yield path :: parents + + override def getOptionsFromFiles(command: String): UIO[List[OptionsFromFile]] = + (for { + cwd <- System.property("user.dir") + home <- System.property("user.home") + commandFile = s".$command" + + pathsFromCWD <- cwd match { + case Some(cwd) => ZIO.attempt(Path.of(cwd)).flatMap(getPathAndParents) + case None => ZIO.succeed(Nil) + } + homePath <- ZIO.foreach(home)(p => ZIO.attempt(Path.of(p))) + allPaths = (pathsFromCWD ::: homePath.toList).distinct + + argsFromFiles <- ZIO.foreach(allPaths) { path => + ZIO.attempt(path.resolve(commandFile)).flatMap(optReadPath) + } + } yield argsFromFiles.flatten) + .catchAllCause(ZIO.logErrorCause(s"Error reading options from files, skipping...", _).as(Nil)) + + } + +} diff --git a/zio-cli/shared/src/main/scala/zio/cli/Options.scala b/zio-cli/shared/src/main/scala/zio/cli/Options.scala index 9d024e52..f23b706a 100644 --- a/zio-cli/shared/src/main/scala/zio/cli/Options.scala +++ b/zio-cli/shared/src/main/scala/zio/cli/Options.scala @@ -297,6 +297,69 @@ object Options extends OptionsPlatformSpecific { .catchAll(e => ZIO.succeed((Some(e), input, Predef.Map.empty))) } + private def mergeFileOptions( + queue: List[FileOptions.OptionsFromFile], + acc: Predef.Map[String, (String, List[String])], + options: List[Options[_] with Input], + conf: CliConfig + ): IO[ValidationError, Predef.Map[String, (String, List[String])]] = + queue match { + case FileOptions.OptionsFromFile(path, args) :: tail => + for { + tuple1 <- matchOptions(args, options, conf) + newMap <- + tuple1 match { + case (Some(e), _, _) => ZIO.fail(e) + case (_, rest, _) if rest.nonEmpty => + ZIO.fail( + ValidationError( + ValidationErrorType.InvalidArgument, + HelpDoc.p("Files can only contain options, not Args") + + HelpDoc.p(s"Found args: ${rest.mkString(", ")}") + ) + ) + case (_, _, map) => ZIO.succeed(map.map { case (k, v) => (k, (path, v)) }) + } + mergedPairs <- + ZIO.foreach((acc.keySet | newMap.keySet).toList.sorted.map(k => (k, acc.get(k), newMap.get(k)))) { + case (k, Some((accPath, _)), Some(newRes @ (newPath, _))) => + ZIO.logDebug(s"option '$k' : options from path '$accPath' overridden by path '$newPath'") *> + ZIO.some(k -> newRes) + case (k, Some(accRes), None) => + ZIO.some(k -> accRes) + case (k, None, Some(newRes)) => + ZIO.some(k -> newRes) + case (_, None, None) => + ZIO.none + } + mergedMap = mergedPairs.flatten.toMap + res <- mergeFileOptions(tail, mergedMap, options, conf) + } yield res + case Nil => + ZIO.succeed(acc) + } + + private def mergeFileAndInputOptions( + inputOptions: Predef.Map[String, List[String]], + fileOptions: Predef.Map[String, (String, List[String])] + ): UIO[Predef.Map[String, List[String]]] = + ZIO + .foreach( + (fileOptions.keySet | inputOptions.keySet).toList.sorted.map(k => (k, fileOptions.get(k), inputOptions.get(k))) + ) { + case (k, Some((accPath, _)), Some(inputRes)) => + ZIO.logDebug(s"option '$k' : options from path '$accPath' overridden by command line") *> + ZIO.some(k -> inputRes) + case (k, Some((accPath, accArgs)), None) => + ZIO.logInfo(s"option '$k' : using options from path '$accPath' - ${accArgs.mkString(", ")}") *> + ZIO.some(k -> accArgs) + case (k, None, Some(newRes)) => + ZIO.some(k -> newRes) + case (_, None, None) => + ZIO.none + } + .map(_.flatten.toMap) + /** * `Options.validate` parses `args` for `options and returns an `Option[ValidationError]`, the leftover arguments and * the constructed value of type `A`. The possible error inside `Option[ValidationError]` would only be triggered if @@ -306,12 +369,15 @@ object Options extends OptionsPlatformSpecific { def validate[A]( options: Options[A], args: List[String], - conf: CliConfig + conf: CliConfig, + fromFiles: List[FileOptions.OptionsFromFile] = Nil ): IO[ValidationError, (Option[ValidationError], List[String], A)] = for { matched <- matchOptions(args, options.flatten, conf) (error, commandArgs, matchedOptions) = matched - a <- options.validate(matchedOptions, conf).mapError(error.getOrElse(_)) + mapFromFiles <- mergeFileOptions(fromFiles.reverse, Predef.Map.empty, options.flatten, conf) + mergedOptions <- mergeFileAndInputOptions(matchedOptions, mapFromFiles) + a <- options.validate(mergedOptions, conf).mapError(error.getOrElse(_)) } yield (error, commandArgs, a) case object Empty extends Options[Unit] with Pipeline { diff --git a/zio-cli/shared/src/test/scala/zio/cli/CommandSpec.scala b/zio-cli/shared/src/test/scala/zio/cli/CommandSpec.scala index 5181ff9f..32b32e5b 100644 --- a/zio-cli/shared/src/test/scala/zio/cli/CommandSpec.scala +++ b/zio-cli/shared/src/test/scala/zio/cli/CommandSpec.scala @@ -27,10 +27,12 @@ object CommandSpec extends ZIOSpecDefault { suite("Toplevel Command Spec")( suite("Command with options followed by args")( test("Should validate successfully") { - assertZIO(Tail.command.parse(List("tail", "-n", "100", "foo.log"), CliConfig.default))( + assertZIO(Tail.command.parse(List("tail", "-n", "100", "foo.log"), CliConfig.default, Nil))( equalTo(CommandDirective.UserDefined(List.empty[String], (BigInt(100), "foo.log"))) ) *> - assertZIO(Ag.command.parse(List("grep", "--after", "2", "--before", "3", "fooBar"), CliConfig.default))( + assertZIO( + Ag.command.parse(List("grep", "--after", "2", "--before", "3", "fooBar"), CliConfig.default) + )( equalTo(CommandDirective.UserDefined(List.empty[String], ((BigInt(2), BigInt(3)), "fooBar"))) ) }, @@ -81,7 +83,7 @@ object CommandSpec extends ZIOSpecDefault { val orElseCommand = Command("remote", Options.Empty, Args.none) | Command("log", Options.Empty, Args.none) - assertZIO(orElseCommand.parse(List("log"), CliConfig.default))( + assertZIO(orElseCommand.parse(List("log"), CliConfig.default, Nil))( equalTo(CommandDirective.UserDefined(Nil, ())) ) } @@ -131,17 +133,17 @@ object CommandSpec extends ZIOSpecDefault { Vector( test("match first sub command without any surplus arguments") { - assertZIO(git.parse(List("git", "remote"), CliConfig.default))( + assertZIO(git.parse(List("git", "remote"), CliConfig.default, Nil))( equalTo(CommandDirective.UserDefined(Nil, ())) ) }, test("match first sub command with a surplus options") { - assertZIO(git.parse(List("git", "remote", "-v"), CliConfig.default))( + assertZIO(git.parse(List("git", "remote", "-v"), CliConfig.default, Nil))( equalTo(CommandDirective.UserDefined(List("-v"), ())) ) }, test("match second sub command without any surplus arguments") { - assertZIO(git.parse(List("git", "log"), CliConfig.default))( + assertZIO(git.parse(List("git", "log"), CliConfig.default, Nil))( equalTo(CommandDirective.UserDefined(Nil, ())) ) }, @@ -167,7 +169,7 @@ object CommandSpec extends ZIOSpecDefault { Vector( test("test sub command with required options and arguments") { - assertZIO(git.parse(List("git", "rebase", "-i", "upstream", "branch"), CliConfig.default))( + assertZIO(git.parse(List("git", "rebase", "-i", "upstream", "branch"), CliConfig.default, Nil))( equalTo(CommandDirective.UserDefined(Nil, ((true, "drop"), ("upstream", "branch")))) ) }, @@ -212,7 +214,7 @@ object CommandSpec extends ZIOSpecDefault { ) test("sub sub command with option and argument")( - assertZIO(command.parse(List("command", "sub", "subsub", "-i", "text"), CliConfig.default))( + assertZIO(command.parse(List("command", "sub", "subsub", "-i", "text"), CliConfig.default, Nil))( equalTo(CommandDirective.UserDefined(Nil, (true, "text"))) ) ) @@ -222,7 +224,7 @@ object CommandSpec extends ZIOSpecDefault { suite("test adding helpdoc to commands")( test("add text helpdoc to Single") { val command = Command("tldr").withHelp("this is some help") - assertZIO(command.parse(List("tldr"), CliConfig.default))( + assertZIO(command.parse(List("tldr"), CliConfig.default, Nil))( equalTo(CommandDirective.UserDefined(Nil, ())) ) }, diff --git a/zio-cli/shared/src/test/scala/zio/cli/FileOptionsOverrideSpec.scala b/zio-cli/shared/src/test/scala/zio/cli/FileOptionsOverrideSpec.scala new file mode 100644 index 00000000..ef92ad30 --- /dev/null +++ b/zio-cli/shared/src/test/scala/zio/cli/FileOptionsOverrideSpec.scala @@ -0,0 +1,75 @@ +package zio.cli + +import zio._ +import zio.internal.stacktracer.SourceLocation +import zio.test._ +import zio.test.Assertion._ + +object FileOptionsOverrideSpec extends ZIOSpecDefault { + + private final case class Const(fileArgs: List[FileOptions.OptionsFromFile]) extends FileOptions { + override def getOptionsFromFiles(command: String): UIO[List[FileOptions.OptionsFromFile]] = ZIO.succeed(fileArgs) + } + + private final case class Result( + arg1: String, + arg2: String, + arg3: String, + arg4: String + ) + + private val options: Options[Result] = + ( + Options.text("arg-1") ++ + Options.text("arg-2") ++ + Options.text("arg-3") ++ + Options.text("arg-4") + ).map((Result.apply _).tupled) + + private val cliApp: CliApp[Any, Nothing, Result] = + CliApp.make( + "test-app", + "v0", + HelpDoc.Span.empty, + Command("cmd", options) + )(ZIO.succeed(_)) + + private def makeTest( + name: String + )(cmdLine: String*)(fromFiles: (String, List[String])*)(exp: Result)(implicit loc: SourceLocation): Spec[Any, Any] = + test(name) { + cliApp + .runWithFileArgs(cmdLine.toList) + .map(assert(_)(isSome(equalTo(exp)))) + .provideLayer(ZLayer.succeed(Const(fromFiles.toList.map { case (p, a) => FileOptions.OptionsFromFile(p, a) }))) + } + + override def spec: Spec[TestEnvironment with Scope, Any] = + suite("FileOptionsOverrideSpec")( + suite("option+param together")( + makeTest("all file args overridden")("--arg-1=arg", "--arg-2=arg", "--arg-3=arg", "--arg-4=arg")( + "/a/b/c/.cmd" -> List("--arg-1=/a/b/c.cmd"), + "/a/b/.cmd" -> List("--arg-2=/a/b.cmd"), + "/a/.cmd" -> List("--arg-3=/a.cmd") + )(Result("arg", "arg", "arg", "arg")), + makeTest("inheritance hierarchy")("--arg-1=arg")( + "/a/b/c/.cmd" -> List("--arg-1=/a/b/c.cmd", "--arg-2=/a/b/c.cmd"), + "/a/b/.cmd" -> List("--arg-1=/a/b.cmd", "--arg-2=/a/b.cmd", "--arg-3=/a/b.cmd"), + "/a/.cmd" -> List("--arg-1=/a.cmd", "--arg-2=/a.cmd", "--arg-3=/a.cmd", "--arg-4=/a.cmd") + )(Result("arg", "/a/b/c.cmd", "/a/b.cmd", "/a.cmd")) + ), + suite("option+param separate")( + makeTest("all file args overridden")("--arg-1", "arg", "--arg-2", "arg", "--arg-3", "arg", "--arg-4", "arg")( + "/a/b/c/.cmd" -> List("--arg-1", "/a/b/c.cmd"), + "/a/b/.cmd" -> List("--arg-2", "/a/b.cmd"), + "/a/.cmd" -> List("--arg-3", "/a.cmd") + )(Result("arg", "arg", "arg", "arg")), + makeTest("inheritance hierarchy")("--arg-1", "arg")( + "/a/b/c/.cmd" -> List("--arg-1", "/a/b/c.cmd", "--arg-2", "/a/b/c.cmd"), + "/a/b/.cmd" -> List("--arg-1", "/a/b.cmd", "--arg-2", "/a/b.cmd", "--arg-3", "/a/b.cmd"), + "/a/.cmd" -> List("--arg-1", "/a.cmd", "--arg-2", "/a.cmd", "--arg-3", "/a.cmd", "--arg-4", "/a.cmd") + )(Result("arg", "/a/b/c.cmd", "/a/b.cmd", "/a.cmd")) + ) + ) + +} diff --git a/zio-cli/shared/src/test/scala/zio/cli/LiveFileOptionsSpecShared.scala b/zio-cli/shared/src/test/scala/zio/cli/LiveFileOptionsSpecShared.scala new file mode 100644 index 00000000..5381e58d --- /dev/null +++ b/zio-cli/shared/src/test/scala/zio/cli/LiveFileOptionsSpecShared.scala @@ -0,0 +1,82 @@ +package zio.cli + +import zio._ +import zio.internal.stacktracer.SourceLocation +import zio.test._ + +import java.nio.file.{Files, Path} + +abstract class LiveFileOptionsSpecShared extends ZIOSpecDefault { + + private val createTempDirectory: RIO[Scope, Path] = + for { + random <- Random.nextUUID + path <- + ZIO.attempt(Files.createTempDirectory(random.toString)).withFinalizer(f => ZIO.attempt(f.toFile.delete()).orDie) + } yield path + + private def resolvePath(path: Path, paths: List[String]): Path = + if (paths.nonEmpty) path.resolve(paths.mkString("/")) + else path + + private def makeTest(name: String)(cwd: List[String], home: List[String])( + writeFiles: (List[String], String)* + )( + exp: (List[String], List[String])* + )(implicit loc: SourceLocation): Spec[Scope, Throwable] = + test(name) { + for { + // setup + dir <- createTempDirectory + _ <- TestSystem.putProperty("user.dir", resolvePath(dir, cwd).toString) + _ <- TestSystem.putProperty("user.home", resolvePath(dir, home).toString) + _ <- ZIO.foreachDiscard(writeFiles) { case (paths, contents) => + val writePath = resolvePath(dir, paths :+ s".$cmd") + val parentFile = writePath.getParent.toFile + ZIO.attempt(parentFile.mkdirs()).unlessZIO(ZIO.attempt(parentFile.exists())) *> + ZIO.writeFile(writePath.toString, contents) + } + + // test + result <- FileOptions.Live.getOptionsFromFiles(cmd) + resolvedExp = exp.toList.map { case (paths, args) => + FileOptions.OptionsFromFile(resolvePath(dir, paths :+ s".$cmd").toString, args) + } + + } yield assertTrue(result == resolvedExp) + } + + private val cmd: String = "command" + + override def spec: Spec[TestEnvironment with Scope, Any] = + suite("FileArgsSpec")( + makeTest("empty")(List("abc", "home"), List("abc", "home"))()(), + makeTest("home in cwd parent path")(List("abc", "home", "d", "e", "f"), List("abc", "home"))( + List("abc", "home", "d") -> "d\nd\n\n", + List("abc", "home", "d", "e") -> "e\ne\n\n", + List("abc", "home", "d", "e", "f") -> "f\nf\n\n", + List("abc", "home") -> "_home_" + )( + List("abc", "home", "d", "e", "f") -> List("f", "f"), + List("abc", "home", "d", "e") -> List("e", "e"), + List("abc", "home", "d") -> List("d", "d"), + List("abc", "home") -> List("_home_") // only appears once + ), + makeTest("home not in cwd parent path")(List("abc", "cwd", "d", "e", "f"), List("abc", "home"))( + List("abc", "cwd", "d") -> "d\nd\n\n", + List("abc", "cwd", "d", "e") -> "e\ne\n\n", + List("abc", "cwd", "d", "e", "f") -> "f\nf\n\n", + List("abc", "home") -> "_home_" + )( + List("abc", "cwd", "d", "e", "f") -> List("f", "f"), + List("abc", "cwd", "d", "e") -> List("e", "e"), + List("abc", "cwd", "d") -> List("d", "d"), + List("abc", "home") -> List("_home_") + ), + makeTest("parent dirs of home are not searched")(Nil, List("abc", "home"))( + List("abc") -> "a\nb" + )( + ) + ) @@ TestAspect.withLiveRandom + +}