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:
parent
6f57f27621
commit
83f3a3ec86
7 changed files with 115 additions and 21 deletions
|
@ -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!");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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())
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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>();
|
||||||
|
|
|
@ -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(),
|
||||||
|
|
|
@ -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");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
29
test/dotnet.Tests/ParserTests/RunParserTests.cs
Normal file
29
test/dotnet.Tests/ParserTests/RunParserTests.cs
Normal 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");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue