mirror of
				https://github.com/Tyrrrz/CliFx.git
				synced 2025-10-25 15:19:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			453 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			453 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Linq;
 | |
| using CliFx.Attributes;
 | |
| using CliFx.Domain;
 | |
| using CliFx.Internal.Extensions;
 | |
| 
 | |
| namespace CliFx.Exceptions
 | |
| {
 | |
|     /// <summary>
 | |
|     /// Domain exception thrown within CliFx.
 | |
|     /// </summary>
 | |
|     public partial class CliFxException : Exception
 | |
|     {
 | |
|         private readonly bool _isMessageSet;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Initializes an instance of <see cref="CliFxException"/>.
 | |
|         /// </summary>
 | |
|         public CliFxException(string? message, Exception? innerException = null)
 | |
|             : base(message, innerException)
 | |
|         {
 | |
|             // Message property has a fallback so it's never empty, hence why we need this check
 | |
|             _isMessageSet = !string.IsNullOrWhiteSpace(message);
 | |
|         }
 | |
| 
 | |
|         /// <inheritdoc />
 | |
|         public override string ToString() => _isMessageSet
 | |
|             ? Message
 | |
|             : base.ToString();
 | |
|     }
 | |
| 
 | |
|     // Internal exceptions
 | |
|     // Provide more diagnostic information here
 | |
|     public partial class CliFxException
 | |
|     {
 | |
|         internal static CliFxException DefaultActivatorFailed(Type type, Exception? innerException = null)
 | |
|         {
 | |
|             var configureActivatorMethodName = $"{nameof(CliApplicationBuilder)}.{nameof(CliApplicationBuilder.UseTypeActivator)}(...)";
 | |
| 
 | |
|             var message = $@"
 | |
| Failed to create an instance of type '{type.FullName}'.
 | |
| The type must have a public parameterless constructor in order to be instantiated by the default activator.
 | |
| 
 | |
| To fix this, either make sure this type has a public parameterless constructor, or configure a custom activator using {configureActivatorMethodName}. 
 | |
| Refer to the readme to learn how to integrate a dependency container of your choice to act as a type activator.";
 | |
| 
 | |
|             return new CliFxException(message.Trim(), innerException);
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException DelegateActivatorReturnedNull(Type type)
 | |
|         {
 | |
|             var message = $@"
 | |
| Failed to create an instance of type '{type.FullName}', received <null> instead.
 | |
| 
 | |
| To fix this, ensure that the provided type activator was configured correctly, as it's not expected to return <null>.
 | |
| If you are using a dependency container, this error may signify that the type wasn't registered.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException InvalidCommandType(Type type)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{type.FullName}' is not a valid command type.
 | |
| 
 | |
| In order to be a valid command type, it must:
 | |
| - Not be an abstract class
 | |
| - Implement {typeof(ICommand).FullName}
 | |
| - Be annotated with {typeof(CommandAttribute).FullName}
 | |
| 
 | |
| If you're experiencing problems, please refer to the readme for a quickstart example.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException NoCommandsDefined()
 | |
|         {
 | |
|             var message = $@"
 | |
| There are no commands configured in the application.
 | |
| 
 | |
| To fix this, ensure that at least one command is added through one of the methods on {nameof(CliApplicationBuilder)}.
 | |
| If you're experiencing problems, please refer to the readme for a quickstart example.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException TooManyDefaultCommands(IReadOnlyList<CommandSchema> invalidCommands)
 | |
|         {
 | |
|             var message = $@"
 | |
| Application configuration is invalid because there are {invalidCommands.Count} default commands:
 | |
| {invalidCommands.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| There can only be one default command (i.e. command with no name) in an application.
 | |
| Other commands must have unique non-empty names that identify them.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException CommandsWithSameName(
 | |
|             string name,
 | |
|             IReadOnlyList<CommandSchema> invalidCommands)
 | |
|         {
 | |
|             var message = $@"
 | |
| Application configuration is invalid because there are {invalidCommands.Count} commands with the same name ('{name}'):
 | |
| {invalidCommands.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Commands must have unique names.
 | |
| Names are not case-sensitive.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException ParametersWithSameOrder(
 | |
|             CommandSchema command,
 | |
|             int order,
 | |
|             IReadOnlyList<CommandParameterSchema> invalidParameters)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidParameters.Count} parameters with the same order ({order}):
 | |
| {invalidParameters.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Parameters must have unique order.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException ParametersWithSameName(
 | |
|             CommandSchema command,
 | |
|             string name,
 | |
|             IReadOnlyList<CommandParameterSchema> invalidParameters)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidParameters.Count} parameters with the same name ('{name}'):
 | |
| {invalidParameters.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Parameters must have unique names to avoid potential confusion in the help text.
 | |
| Names are not case-sensitive.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException TooManyNonScalarParameters(
 | |
|             CommandSchema command,
 | |
|             IReadOnlyList<CommandParameterSchema> invalidParameters)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidParameters.Count} non-scalar parameters:
 | |
| {invalidParameters.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Non-scalar parameter is such that is bound from more than one value (e.g. array).
 | |
| Only one parameter in a command may be non-scalar and it must be the last one in order.
 | |
| 
 | |
| If it's not feasible to fit into these constraints, consider using options instead as they don't have these limitations.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException NonLastNonScalarParameter(
 | |
|             CommandSchema command,
 | |
|             CommandParameterSchema invalidParameter)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains a non-scalar parameter which is not the last in order:
 | |
| {invalidParameter}
 | |
| 
 | |
| Non-scalar parameter is such that is bound from more than one value (e.g. array).
 | |
| Only one parameter in a command may be non-scalar and it must be the last one in order.
 | |
| 
 | |
| If it's not feasible to fit into these constraints, consider using options instead as they don't have these limitations.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException ParametersWithInvalidConverters(
 | |
|             CommandSchema command,
 | |
|             IReadOnlyList<CommandParameterSchema> invalidParameters)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidParameters.Count} parameter(s) with invalid converters:
 | |
| {invalidParameters.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Specified converter must implement {typeof(IArgumentValueConverter).FullName}.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException ParametersWithInvalidValidators(
 | |
|             CommandSchema command,
 | |
|             IReadOnlyList<CommandParameterSchema> invalidParameters)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidParameters.Count} parameter(s) with invalid value validators:
 | |
| {invalidParameters.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Specified validator(s) must inherit from {typeof(ArgumentValueValidator<>).FullName}.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException OptionsWithNoName(
 | |
|             CommandSchema command,
 | |
|             IReadOnlyList<CommandOptionSchema> invalidOptions)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains one or more options without a name:
 | |
| {invalidOptions.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Options must have either a name or a short name or both.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException OptionsWithInvalidLengthName(
 | |
|             CommandSchema command,
 | |
|             IReadOnlyList<CommandOptionSchema> invalidOptions)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains one or more options whose names are too short:
 | |
| {invalidOptions.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Option names must be at least 2 characters long to avoid confusion with short names.
 | |
| If you intended to set the short name instead, use the attribute overload that accepts a char.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException OptionsWithSameName(
 | |
|             CommandSchema command,
 | |
|             string name,
 | |
|             IReadOnlyList<CommandOptionSchema> invalidOptions)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidOptions.Count} options with the same name ('{name}'):
 | |
| {invalidOptions.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Options must have unique names.
 | |
| Names are not case-sensitive.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException OptionsWithSameShortName(
 | |
|             CommandSchema command,
 | |
|             char shortName,
 | |
|             IReadOnlyList<CommandOptionSchema> invalidOptions)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidOptions.Count} options with the same short name ('{shortName}'):
 | |
| {invalidOptions.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Options must have unique short names.
 | |
| Short names are case-sensitive (i.e. 'a' and 'A' are different short names).";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException OptionsWithSameEnvironmentVariableName(
 | |
|             CommandSchema command,
 | |
|             string environmentVariableName,
 | |
|             IReadOnlyList<CommandOptionSchema> invalidOptions)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidOptions.Count} options with the same fallback environment variable name ('{environmentVariableName}'):
 | |
| {invalidOptions.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Options cannot share the same environment variable as a fallback.
 | |
| Environment variable names are not case-sensitive.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException OptionsWithInvalidConverters(
 | |
|             CommandSchema command,
 | |
|             IReadOnlyList<CommandOptionSchema> invalidOptions)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidOptions.Count} option(s) with invalid converters:
 | |
| {invalidOptions.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Specified converter must implement {typeof(IArgumentValueConverter).FullName}.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException OptionsWithInvalidValidators(
 | |
|             CommandSchema command,
 | |
|             IReadOnlyList<CommandOptionSchema> invalidOptions)
 | |
|         {
 | |
|             var message = $@"
 | |
| Command '{command.Type.FullName}' is invalid because it contains {invalidOptions.Count} option(s) with invalid validators:
 | |
| {invalidOptions.JoinToString(Environment.NewLine)}
 | |
| 
 | |
| Specified validators must inherit from {typeof(IArgumentValueValidator).FullName}.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // End-user-facing exceptions
 | |
|     // Avoid internal details and fix recommendations here
 | |
|     public partial class CliFxException
 | |
|     {
 | |
|         internal static CliFxException CannotConvertMultipleValuesToNonScalar(
 | |
|             CommandParameterSchema parameter,
 | |
|             IReadOnlyList<string> values)
 | |
|         {
 | |
|             var message = $@"
 | |
| Parameter {parameter.GetUserFacingDisplayString()} expects a single value, but provided with multiple:
 | |
| {values.Select(v => v.Quote()).JoinToString(" ")}";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException CannotConvertMultipleValuesToNonScalar(
 | |
|             CommandOptionSchema option,
 | |
|             IReadOnlyList<string> values)
 | |
|         {
 | |
|             var message = $@"
 | |
| Option {option.GetUserFacingDisplayString()} expects a single value, but provided with multiple:
 | |
| {values.Select(v => v.Quote()).JoinToString(" ")}";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException CannotConvertMultipleValuesToNonScalar(
 | |
|             CommandArgumentSchema argument,
 | |
|             IReadOnlyList<string> values) => argument switch
 | |
|         {
 | |
|             CommandParameterSchema parameter => CannotConvertMultipleValuesToNonScalar(parameter, values),
 | |
|             CommandOptionSchema option => CannotConvertMultipleValuesToNonScalar(option, values),
 | |
|             _ => throw new ArgumentOutOfRangeException(nameof(argument))
 | |
|         };
 | |
| 
 | |
|         internal static CliFxException CannotConvertToType(
 | |
|             CommandParameterSchema parameter,
 | |
|             string? value,
 | |
|             Type type,
 | |
|             Exception? innerException = null)
 | |
|         {
 | |
|             var message = $@"
 | |
| Can't convert value ""{value ?? "<null>"}"" to type '{type.Name}' for parameter {parameter.GetUserFacingDisplayString()}.
 | |
| {innerException?.Message ?? "This type is not supported."}";
 | |
| 
 | |
|             return new CliFxException(message.Trim(), innerException);
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException CannotConvertToType(
 | |
|             CommandOptionSchema option,
 | |
|             string? value,
 | |
|             Type type,
 | |
|             Exception? innerException = null)
 | |
|         {
 | |
|             var message = $@"
 | |
| Can't convert value ""{value ?? "<null>"}"" to type '{type.Name}' for option {option.GetUserFacingDisplayString()}.
 | |
| {innerException?.Message ?? "This type is not supported."}";
 | |
| 
 | |
|             return new CliFxException(message.Trim(), innerException);
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException CannotConvertToType(
 | |
|             CommandArgumentSchema argument,
 | |
|             string? value,
 | |
|             Type type,
 | |
|             Exception? innerException = null) => argument switch
 | |
|         {
 | |
|             CommandParameterSchema parameter => CannotConvertToType(parameter, value, type, innerException),
 | |
|             CommandOptionSchema option => CannotConvertToType(option, value, type, innerException),
 | |
|             _ => throw new ArgumentOutOfRangeException(nameof(argument))
 | |
|         };
 | |
| 
 | |
|         internal static CliFxException CannotConvertNonScalar(
 | |
|             CommandParameterSchema parameter,
 | |
|             IReadOnlyList<string> values,
 | |
|             Type type)
 | |
|         {
 | |
|             var message = $@"
 | |
| Can't convert provided values to type '{type.Name}' for parameter {parameter.GetUserFacingDisplayString()}:
 | |
| {values.Select(v => v.Quote()).JoinToString(" ")}
 | |
| 
 | |
| Target type is not assignable from array and doesn't have a public constructor that takes an array.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException CannotConvertNonScalar(
 | |
|             CommandOptionSchema option,
 | |
|             IReadOnlyList<string> values,
 | |
|             Type type)
 | |
|         {
 | |
|             var message = $@"
 | |
| Can't convert provided values to type '{type.Name}' for option {option.GetUserFacingDisplayString()}:
 | |
| {values.Select(v => v.Quote()).JoinToString(" ")}
 | |
| 
 | |
| Target type is not assignable from array and doesn't have a public constructor that takes an array.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException CannotConvertNonScalar(
 | |
|             CommandArgumentSchema argument,
 | |
|             IReadOnlyList<string> values,
 | |
|             Type type) => argument switch
 | |
|         {
 | |
|             CommandParameterSchema parameter => CannotConvertNonScalar(parameter, values, type),
 | |
|             CommandOptionSchema option => CannotConvertNonScalar(option, values, type),
 | |
|             _ => throw new ArgumentOutOfRangeException(nameof(argument))
 | |
|         };
 | |
| 
 | |
|         internal static CliFxException ParameterNotSet(CommandParameterSchema parameter)
 | |
|         {
 | |
|             var message = $@"
 | |
| Missing value for parameter {parameter.GetUserFacingDisplayString()}.";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException RequiredOptionsNotSet(IReadOnlyList<CommandOptionSchema> options)
 | |
|         {
 | |
|             var message = $@"
 | |
| Missing values for one or more required options:
 | |
| {options.Select(o => o.GetUserFacingDisplayString()).JoinToString(Environment.NewLine)}";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException UnrecognizedParametersProvided(IReadOnlyList<CommandParameterInput> parameterInputs)
 | |
|         {
 | |
|             var message = $@"
 | |
| Unrecognized parameters provided:
 | |
| {parameterInputs.Select(p => p.Value).JoinToString(Environment.NewLine)}";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException UnrecognizedOptionsProvided(IReadOnlyList<CommandOptionInput> optionInputs)
 | |
|         {
 | |
|             var message = $@"
 | |
| Unrecognized options provided:
 | |
| {optionInputs.Select(o => o.GetRawAlias()).JoinToString(Environment.NewLine)}";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
| 
 | |
|         internal static CliFxException ValueValidationFailed(CommandArgumentSchema argument, IEnumerable<string> errors)
 | |
|         {
 | |
|             var message = $@"
 | |
| The validation of the provided value for {argument.Property!.Name} is failed because: {errors.JoinToString(Environment.NewLine)}";
 | |
| 
 | |
|             return new CliFxException(message.Trim());
 | |
|         }
 | |
|     }
 | |
| } |