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

Minor token improvements #2080

Closed
wants to merge 3 commits 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
Original file line number Diff line number Diff line change
Expand Up @@ -374,9 +374,8 @@ System.CommandLine.Parsing
public T GetValue<T>(Argument<T> argument)
public T GetValue<T>(Option<T> option)
public class Token, System.IEquatable<Token>
public static System.Boolean op_Equality(Token left, Token right)
public static System.Boolean op_Inequality(Token left, Token right)
.ctor(System.String value, TokenType type, System.CommandLine.Symbol symbol)
public System.CommandLine.Symbol Symbol { get; }
public TokenType Type { get; }
public System.String Value { get; }
public System.Boolean Equals(System.Object obj)
Expand Down
53 changes: 53 additions & 0 deletions src/System.CommandLine.Tests/TokenTests.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
using FluentAssertions;
using System.CommandLine.Parsing;
using Xunit;

namespace System.CommandLine.Tests
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are we leveraging file scoped namespaces?
I see at least one file was converted to use them by @jonsequitur in #2226

{
public class TokenTests
{
[Fact]
public void Tokens_are_equal_when_they_use_same_value_type_and_symbol()
{
Option<int> count = new ("--count");
Token token = new (count.Name, TokenType.Option, count);
Token same = new (count.Name, TokenType.Option, count);

token.Equals(same).Should().BeTrue();
token.Equals((object)same).Should().BeTrue();
}

[Fact]
public void Tokens_are_not_equal_when_they_do_not_use_same_value_type_and_symbol()
{
Option<int> symbol = new("--count");
Option<int> symbolWithSameName = new("--count");

Token token = new(symbol.Name, TokenType.Option, symbol);
Token differentValue = new("different", TokenType.Option, symbol);
Token differentType = new(symbol.Name, TokenType.Argument, symbol);
Token differentSymbol = new(symbol.Name, TokenType.Option, symbolWithSameName);

Assert(token, differentValue);
Assert(token, differentType);
Assert(token, differentSymbol);
Assert(token, null);

static void Assert(Token token, Token different)
{
token.Equals(different).Should().BeFalse();
token.Equals((object)different).Should().BeFalse();
}
}

[Fact]
public void Symbol_property_returns_value_provided_in_constructor()
{
Option<int> symbol = new("--count");

Token token = new(symbol.Name, TokenType.Option, symbol);

token.Symbol.Should().Be(symbol);
}
}
}
18 changes: 1 addition & 17 deletions src/System.CommandLine/Parsing/Token.cs
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ internal Token(string? value, TokenType type, Symbol? symbol, int position)
/// <summary>
/// The Symbol represented by the token (if any).
/// </summary>
internal Symbol? Symbol { get; set; }
public Symbol? Symbol { get; internal set; }
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should test that this is correctly reassigned when OnlyTake is used.


/// <inheritdoc />
public override bool Equals(object? obj) => Equals(obj as Token);
Expand All @@ -59,21 +59,5 @@ internal Token(string? value, TokenType type, Symbol? symbol, int position)

/// <inheritdoc />
public override string ToString() => Value;

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Standard guidance (https://learn.microsoft.com/en-us/dotnet/api/system.iequatable-1?view=netstandard-2.0#notes-to-implementers) does suggest we should keep the equality operators for consistency, even if we aren't using them at the moment.

From a personal perspective, I like having the equality operators available since it makes the code simpler when dealing with nullable types, and using the equality operators helps avoid (via compilation errors) accidentally making typos that result in calling type1.Equals(type2) (calling into Type1.Equals(object), always returning false).

/// <summary>
/// Checks if two specified <see cref="Token"/> instances have the same value.
/// </summary>
/// <param name="left">The first <see cref="Token"/>.</param>
/// <param name="right">The second <see cref="Token"/>.</param>
/// <returns><see langword="true" /> if the objects are equal.</returns>
public static bool operator ==(Token? left, Token? right) => left is null ? right is null : left.Equals(right);

/// <summary>
/// Checks if two specified <see cref="Token"/> instances have different values.
/// </summary>
/// <param name="left">The first <see cref="Token"/>.</param>
/// <param name="right">The second <see cref="Token"/>.</param>
/// <returns><see langword="true" /> if the objects are not equal.</returns>
public static bool operator !=(Token? left, Token? right) => left is null ? right is not null : !left.Equals(right);
}
}