Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Issue 377 proposed fix #894

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions src/CommandLine/Core/TypeConverter.cs
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ private static Maybe<object> ChangeTypeSequence(IEnumerable<string> values, Type
var type =
conversionType.GetTypeInfo()
.GetGenericArguments()
.FailIfMoreThanoneElement(new InvalidOperationException("Unsupported sequence type - non scalar properties should be sequence of type IEnumerable<T>."))
.SingleOrDefault()
.ToMaybe()
.FromJustOrFail(
Expand Down
27 changes: 26 additions & 1 deletion src/CommandLine/Infrastructure/CSharpx/EnumerableExtensions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -459,5 +459,30 @@ public static IEnumerable<string> FlattenOnce(this IEnumerable<string> source)
}
}
}

/// <summary>
/// Throws the specified exception if the <c>IEnumerable</c> contains more than one element.
/// </summary>
/// <param name="source">The IEnumerable this extension method is called upon</param>
/// <param name="exceptionToThrow">The exception to throw if the sequence has more than one element</param>
/// <remarks>
/// It is not recommended to use this extension method without a custom exception
/// one could simply use the default Linq .Single() and .SingleOrDefault() methods.
/// </remarks>
public static IEnumerable<T> FailIfMoreThanoneElement<T>(this IEnumerable<T> source, Exception exceptionToThrow = null)
{
if (source == null) throw new ArgumentNullException(nameof(source));

if (source.Take(2).Count() > 1)
{
if (exceptionToThrow != null)
throw exceptionToThrow;

// A desidered exception has not been specified - in this case a default error message is used
throw new InvalidOperationException("Sequence contains more than one elemen");
}

return source;
}
}
}
}
74 changes: 74 additions & 0 deletions tests/CommandLine.Tests/Unit/Issue377Tests.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
using CommandLine.Text;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using static CommandLine.Tests.Unit.Issue389Tests;
using Xunit;
using FluentAssertions;

namespace CommandLine.Tests.Unit
{
//Reference: PR# 377
public class Issue377Tests
{

// Test the normal behavior of the library
[Fact]
public void Test_Read_File_List_With_IEnumerable_And_Type()
{
ParserResult<Options_IEnumerable_With_Type> parsedOptions = Parser.Default.ParseArguments<Options_IEnumerable_With_Type>(new string[] { "--read", "file1", "file2" });
parsedOptions.Tag.Should().Be(ParserResultType.Parsed);
parsedOptions.Value.Should().NotBeNull();
parsedOptions.Value.InputFiles.Should().HaveCount(2);
parsedOptions.Value.InputFiles.First().Should().Be("file1");
parsedOptions.Value.InputFiles.Last().Should().Be("file2");
}


// Test the normal behavior of the library
[Fact]
public void Test_Read_File_List_With_IEnumerable_And_Without_Type()
{
Action parseUnsupportedType = () => Parser.Default.ParseArguments<Options_IEnumerable_Without_Type>(new string[] { "--read", "file1", "file2" });
Assert.Throws<InvalidOperationException>(parseUnsupportedType);
}

// Tests the behaviour of the library with an unsupported type (i.e. a dictionary)
[Fact]
public void Test_Read_File_List_With_IDictionary()
{
Action parseUnsupportedType = () => Parser.Default.ParseArguments<Options_Dictionary>(new string[] { "--read", "file1", "file2" });
Assert.Throws<InvalidOperationException>(parseUnsupportedType);
}

// Options with IEnumerable
internal class Options_IEnumerable_With_Type
{

[Option('r', "read", Required = true, HelpText = "Input files to be processed.")]
public IEnumerable<string> InputFiles { get; set; }

}

// Options with IEnumerable
internal class Options_IEnumerable_Without_Type
{

[Option('r', "read", Required = true, HelpText = "Input files to be processed.")]
public IEnumerable InputFiles { get; set; }

}

// Options with unsupported IDictionary type
internal class Options_Dictionary
{

[Option('r', "read", Required = true, HelpText = "Input files to be processed.")]
public IDictionary<string, string> InputFiles { get; set; }

}
}
}