Fix dotnet run double dash passing arguments

When run “dotnet run -- foo”, foo should be the argument passed to the
subject app. After replacing the original parser, dotnet-run did not
utilize the “unparsedtoken” of the parsed result.

To append unparsedtoken to RunCommand’s argument is not straight
forward. RunCommand has an “immutable constructor”, which is a good
thing, so I made update RunCommand’s argument following the immutable
pattern -- create a new object with the original field but only change
the arguments. I also made these filed private set.
This commit is contained in:
William Li 2017-04-12 16:03:45 -07:00
parent 6f57f27621
commit 83f3a3ec86
7 changed files with 115 additions and 21 deletions

View file

@ -9,6 +9,10 @@ namespace MSBuildTestApp
{ {
public static void Main(string[] args) public static void Main(string[] args)
{ {
if (args.Length > 0)
{
Console.WriteLine("echo args:"+ String.Join(";", args));
}
Console.WriteLine("Hello World!"); Console.WriteLine("Hello World!");
} }
} }

View file

@ -1,4 +1,4 @@
using System; using System;
using System.Linq; using System.Linq;
using Microsoft.DotNet.Cli.CommandLine; using Microsoft.DotNet.Cli.CommandLine;
using Microsoft.DotNet.Cli.Utils; using Microsoft.DotNet.Cli.Utils;
@ -24,6 +24,11 @@ namespace Microsoft.DotNet.Cli
Console.WriteLine(result.Diagram()); Console.WriteLine(result.Diagram());
if (result.UnparsedTokens.Any()) {
Console.WriteLine("Unparsed Tokens: ");
Console.WriteLine(string.Join(" ", (result.UnparsedTokens)));
}
var optionValuesToBeForwarded = result.AppliedCommand() var optionValuesToBeForwarded = result.AppliedCommand()
.OptionValuesToBeForwarded(); .OptionValuesToBeForwarded();
if (optionValuesToBeForwarded.Any()) if (optionValuesToBeForwarded.Any())

View file

@ -1,8 +1,8 @@
// Copyright (c) .NET Foundation and contributors. All rights reserved. // Copyright (c) .NET Foundation and contributors. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information. // Licensed under the MIT license. See LICENSE file in the project root for full license information.
using System; using System.Collections.Generic;
using Microsoft.DotNet.Cli.CommandLine; using System.Linq;
using Microsoft.DotNet.Cli.Utils; using Microsoft.DotNet.Cli.Utils;
using Microsoft.DotNet.Cli; using Microsoft.DotNet.Cli;
using Parser = Microsoft.DotNet.Cli.Parser; using Parser = Microsoft.DotNet.Cli.Parser;
@ -13,13 +13,22 @@ namespace Microsoft.DotNet.Tools.Run
{ {
public static RunCommand FromArgs(string[] args, string msbuildPath = null) public static RunCommand FromArgs(string[] args, string msbuildPath = null)
{ {
var parser = Parser.Instance; var result = Parser.Instance.ParseFrom("dotnet run", args);
var result = parser.ParseFrom("dotnet run", args);
result.ShowHelpOrErrorIfAppropriate(); result.ShowHelpOrErrorIfAppropriate();
return result["dotnet"]["run"].Value<RunCommand>(); var runCommand = result["dotnet"]["run"].Value<RunCommand>();
return IncludingArgumentsAfterDoubleDash(runCommand, result.UnparsedTokens);
}
private static RunCommand IncludingArgumentsAfterDoubleDash(
RunCommand runCommand,
IEnumerable<string> unparsedTokens)
{
return runCommand.MakeNewWithReplaced(
args: runCommand.Args
.Concat(unparsedTokens)
.ToList());
} }
public static int Run(string[] args) public static int Run(string[] args)
@ -27,7 +36,7 @@ namespace Microsoft.DotNet.Tools.Run
DebugHelper.HandleDebugSwitch(ref args); DebugHelper.HandleDebugSwitch(ref args);
RunCommand cmd; RunCommand cmd;
try try
{ {
cmd = FromArgs(args); cmd = FromArgs(args);

View file

@ -13,11 +13,11 @@ namespace Microsoft.DotNet.Tools.Run
{ {
public partial class RunCommand public partial class RunCommand
{ {
public string Configuration { get; set; } public string Configuration { get; private set; }
public string Framework { get; set; } public string Framework { get; private set; }
public bool NoBuild { get; set; } public bool NoBuild { get; private set; }
public string Project { get; set; } public string Project { get; private set; }
public IReadOnlyCollection<string> Args { get; set; } public IReadOnlyCollection<string> Args { get; private set; }
private List<string> _args; private List<string> _args;
private bool ShouldBuild => !NoBuild; private bool ShouldBuild => !NoBuild;
@ -38,6 +38,34 @@ namespace Microsoft.DotNet.Tools.Run
.ExitCode; .ExitCode;
} }
public RunCommand(string configuration,
string framework,
bool noBuild,
string project,
IReadOnlyCollection<string> args)
{
Configuration = configuration;
Framework = framework;
NoBuild = noBuild;
Project = project;
Args = args;
}
public RunCommand MakeNewWithReplaced(string configuration = null,
string framework = null,
bool? noBuild = null,
string project = null,
IReadOnlyCollection<string> args = null)
{
return new RunCommand(
configuration ?? this.Configuration,
framework ?? this.Framework,
noBuild ?? this.NoBuild,
project ?? this.Project,
args ?? this.Args
);
}
private void EnsureProjectIsBuilt() private void EnsureProjectIsBuilt()
{ {
List<string> buildArgs = new List<string>(); List<string> buildArgs = new List<string>();

View file

@ -15,14 +15,14 @@ namespace Microsoft.DotNet.Cli
LocalizableStrings.AppFullName, LocalizableStrings.AppFullName,
treatUnmatchedTokensAsErrors: false, treatUnmatchedTokensAsErrors: false,
arguments: Accept.ZeroOrMoreArguments() arguments: Accept.ZeroOrMoreArguments()
.MaterializeAs(o => new RunCommand .MaterializeAs(o => new RunCommand
{ (
Configuration = o.SingleArgumentOrDefault("--configuration"), configuration: o.SingleArgumentOrDefault("--configuration"),
Framework = o.SingleArgumentOrDefault("--framework"), framework: o.SingleArgumentOrDefault("--framework"),
NoBuild = o.HasOption("--no-build"), noBuild: o.HasOption("--no-build"),
Project = o.SingleArgumentOrDefault("--project"), project: o.SingleArgumentOrDefault("--project"),
Args = o.Arguments args: o.Arguments
}), )),
options: new[] options: new[]
{ {
CommonOptions.HelpOption(), CommonOptions.HelpOption(),

View file

@ -181,5 +181,24 @@ namespace Microsoft.DotNet.Cli.Run.Tests
.Should().Fail() .Should().Fail()
.And.HaveStdErrContaining("--framework"); .And.HaveStdErrContaining("--framework");
} }
[Fact]
public void ItCanPassArgumentsToSubjectAppByDoubleDash()
{
const string testAppName = "MSBuildTestApp";
var testInstance = TestAssets.Get(testAppName)
.CreateInstance()
.WithSourceFiles()
.WithRestoreFiles();
var testProjectDirectory = testInstance.Root.FullName;
new RunCommand()
.WithWorkingDirectory(testProjectDirectory)
.ExecuteWithCapturedOutput("-- foo bar baz")
.Should()
.Pass()
.And.HaveStdOutContaining("echo args:foo;bar;baz");
}
} }
} }

View file

@ -0,0 +1,29 @@
// Copyright (c) .NET Foundation and contributors. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
using System.Linq;
using FluentAssertions;
using Microsoft.DotNet.Tools.Run;
using Xunit;
using Xunit.Abstractions;
using System;
namespace Microsoft.DotNet.Tests.ParserTests
{
public class RunParserTests
{
public RunParserTests(ITestOutputHelper output)
{
this.output = output;
}
private readonly ITestOutputHelper output;
[Fact]
public void RunParserCanGetArguementFromDoubleDash()
{
var runCommand = RunCommand.FromArgs(new[]{ "--", "foo" });
runCommand.Args.Single().Should().Be("foo");
}
}
}