Pasar comando desde CMD a aplicación de consola C#

Iniciado por HCK., 8 Diciembre 2015, 19:51 PM

0 Miembros y 1 Visitante están viendo este tema.

HCK.

Hola chic@s, llevo buscando una duda por internet desde hace tiempo, y tras no encontrar información (tal vez por no saber buscarlo con el tecnicismo exacto, o porque apenas hay información sobre mi duda...) recurro a vosotros haber si me podeis hechar una mano.  :)

Quería saber si es posible hacer interactuar a una aplicación de consola programada en C# con el "símbolo de sistema"... Es decir, por ejemplo:

Desde la consola de Windows llamo a mi aplicación con un comando en concreto que tenga programado y esperar respuesta desde la misma shell de Windows....
miprograma.exe --show "Hello World"

Y que desde la misma consola de Windows, salga la respuesta programada a ese comando de la aplicación de C#.

Gracias  :P, un saludo.

Eleкtro

#1
Para obtener los argumentos que se han pasado por linea de comandos, puedes utilizar el siguiente método:

Para "enviar una respuesta" lo harías escribiendo en el buffer de la consola, con el siguiente método:
(o Console.Write)

Al finalizar la ejecución de la aplicación no olvides enviar un código de salida apropiado utilizando el siguiente método:




Para parsear la sintaxis de los argumento en cuestión eso siempre es trabajo del programador, mediante la manipulación de string (un split, un substring, etc).

Citarmiprograma.exe --show "Hello World"

En ese ejemplo que has mostrado tienes que comprobar que el primer argumento sea un identificador (--cosa) y que exista un string adicional, y que ese string adicional no sea otro identificador sino el valor del parámetro que esperas (ya que el usuario se puede equivocar, y eso requiere un control de errores);
Yo te sugiero utilizar una sintaxis mucho más cómoda y apropiada para el parsing de argumentos command-line (tomando como ejemplo el diseño de algunas aplicaciones de la SDK de Microsoft):
Aplicación.exe /Switch="Valor"
o si lo prefieres así...:
Aplicación.exe --Switch="Valor"

De esa manera lo tienes todo en un único string o argumento facilmente delimitable mediante el símbolo "=", evitando la necesidad de parsear/evaluar un argumento adicional.

Saludos.








HCK.

 :D HoW, eres un máquina. La verdad que no sabia ni por donde empezar, y a pesar de buscarlo con mis palabras, no conseguia encontrar nada. :D :D :D

Muchísimas gracias Elektro :), por cierto, felicitarte por la API que has publicado hace poco, muy buena, y por lo del magnetismo para las ventanas.

Un saludo crack!

Eleкtro

Cita de: HCK. en  8 Diciembre 2015, 22:19 PMLa verdad que no sabia ni por donde empezar, y a pesar de buscarlo con mis palabras, no conseguia encontrar nada. :D :D :D

Desde mi ignorancia me atrevo a decir que probablemente no encuentras información al respecto por que la intentas buscar en Español.
( excepto si utilizases el buscador de la MSDN en Español, claro está. )

Digo yo que una simple búsqueda en Google por palabras clave como "C# read console arguments" te habría sobrado para salir de dudas, y como ves usando términos que ya conocías.




Cita de: HCK. en  8 Diciembre 2015, 22:19 PMMuchísimas gracias Elektro :), por cierto, felicitarte por la API que has publicado hace poco, muy buena, y por lo del magnetismo para las ventanas.

Gracias a ti por el comentario.

Venga, te lo has ganado, no quería ponertelo del todo facl en la parte más importante pero por si te sirve de algo te mostraré como suelo parsear los argumentos (argumentos que sigan las reglas de sintaxis que te comenté en el otro comentario. )

Primero te comentaré la manera más facil (o vaga); lo que yo suelo hacer es declarar una variable por cada argumento en el código fuente (ya que se supone que cada argumento que el usuario pase debe corresponder a un objeto/variable cuyo valor sea asignado en el código fuente),
y entonces con este método parseo los argumentos y le asigno los valores a esas variables:

( Esto es un código traducido al vuelo de Vb.Net a C# )

Código (csharp) [Seleccionar]
private string variable;

private void ParseArguments(IEnumerable<string> args = null) {

foreach (string arg in args) {
switch (true) {
case arg.StartsWith("/?"):
                                // Print Help.
break;

case arg.StartsWith("/Switch1=", StringComparison.OrdinalIgnoreCase):
variable = arg.Substring(arg.IndexOf("=") + 1);
break;
}

}
}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================


Es una solución simple pero efectiva hasta cierto punto ya que no hay controles de errores, pero siempre puedes adaptarlo para que sea tan sofisticado que quieras que sea.

Saludos!








Eleкtro

#4
Ahora vamos con la manera tediosa, pero más eficiente y sobre todo organizada.

Primero de todo, creamos un Type para representar un parámetro command-line.

Lo que deberiamos tener en cuenta en el diseño del modelo es al menos lo siguiente:

  • El nombre del parámetro
  • El símbolo que separa al nombre del parámetro del valor que indique el usuario en el argumento.
  • El valor por defecto del parámetro (para parámetros que son opcionales, o también para argumentos vacíos. )
  • El valor que le da el usuario (valor que parsearemos en los argumentos de la aplicación)
  • Un flag que indique si este parámetro es opcional o no. Si no es opcional entonces deberiamos mostrar un error cuando el usuario no usa este parámetro.

( A la lista que cada cual le añada otros factores que sean importantes para sus necesidades, como pro ejemplo el orden en el que se reciben los parámetros en los argumentos. )

Ahí va (lo he convertido a C# pero no he testeado la sintaxis) :
Código (csharp) [Seleccionar]
// ***********************************************************************
// Author   : Elektro
// Modified : 09-December-2015
// ***********************************************************************

using Microsoft.VisualBasic;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Data;
using System.Diagnostics;

namespace Types
{

#region " Commandline Parameter (Of T) "

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Represents a Commandline Parameter.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <typeparam name="T">
/// The type of value that the parameter takes.
/// </typeparam>
/// ----------------------------------------------------------------------------------------------------
public class CommandlineParameter<T>
{

#region " Properties "

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Gets or sets a value indicating whether this parameter is required for the application.
/// <para></para>
/// A value of <see langword="true"/> means the user needs to assign a value for this parameter.
/// <para></para>
/// A value of <see langword="false"/> means this is an optional parameter so no matter if the user sets a custom value.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <value>
/// <see langword="true"/> if this parameter is required for the application; otherwise, <see langword="false"/>.
/// </value>
/// ----------------------------------------------------------------------------------------------------
public bool IsRequired { get; set; }

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Gets or sets the parameter name.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <value>
/// The parameter name.
/// </value>
/// ----------------------------------------------------------------------------------------------------
public string Name {
[DebuggerStepThrough()]
get { return this.nameB; }
[DebuggerStepThrough()]
set { this.EvaluateName(value); }
}
/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// ( Backing Field )
/// The parameter name.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
private string nameB;

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Gets or sets the parameter separator.
/// <para></para>
/// This character separates the parameter from the value in the argument.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <value>
/// The parameter separator.
/// </value>
/// ----------------------------------------------------------------------------------------------------
public char Separator {
[DebuggerStepThrough()]
get { return this.separatorB; }
[DebuggerStepThrough()]
set { this.EvaluateSeparator(value); }
}
/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// ( Backing Field )
/// The parameter separator.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
private char separatorB;

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Gets or sets the parameter value.
/// <para></para>
/// This value should be initially <see langword="Nothing"/> before parsing the commandline arguments of the application;
/// <para></para>
/// the value of the parameter should be assigned by the end-user when passing an argument to the application.
/// <para></para>
/// To set a default value for this parameter, use <see cref="CommandlineParameter(Of T).DefaultValue"/> property instead.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <value>
/// The parameter value.
/// </value>
/// ----------------------------------------------------------------------------------------------------
public T Value { get; set; }

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Gets or sets the default parameter value.
/// <para></para>
/// This value should be take into account if, after parsing the commandline arguments of the application,
/// <see cref="CommandlineParameter(Of T).Value"/> is <see langword="Nothing"/>,
/// meaning that the end-user didn't assigned any custom value to this parameter.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <value>
/// The default parameter value.
/// </value>
/// ----------------------------------------------------------------------------------------------------
public T DefaultValue { get; set; }

#endregion

#region " Operator Overloading "

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Performs an implicit conversion from <see cref="CommandlineParameter(Of T)"/> to <see cref="CommandlineParameter"/>.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <param name="param">
/// The <see cref="CommandlineParameter(Of T)"/> to convert.
/// </param>
/// ----------------------------------------------------------------------------------------------------
/// <returns>
/// The result of the conversion.
/// </returns>
/// ----------------------------------------------------------------------------------------------------
public static implicit operator CommandlineParameter(CommandlineParameter<T> param)
{

return new CommandlineParameter {
Name = param.Name,
Separator = param.Separator,
DefaultValue = param.DefaultValue,
Value = param.Value,
IsRequired = param.IsRequired
};

}

#endregion

#region " Private Methods "

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Evaluates an attempt to assign the parameter name.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <param name="name">
/// The parameter name.
/// </param>
/// ----------------------------------------------------------------------------------------------------
/// <exception cref="ArgumentException">
/// The parameter name cannot contain the separator character.;name
/// </exception>
/// ----------------------------------------------------------------------------------------------------

protected virtual void EvaluateName(string name)
{
if (!(this.separatorB.Equals(null)) && (name.Contains(this.separatorB))) {
throw new ArgumentException(message: "The parameter name cannot contain the separator character.", paramName: "name");

} else {
this.nameB = name;

}

}

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Evaluates an attempt to assign the parameter separator.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <param name="separator">
/// The parameter separator.
/// </param>
/// ----------------------------------------------------------------------------------------------------
/// <exception cref="ArgumentException">
/// The parameter separator cannot be any character contained in the parameter name.;separator
/// </exception>
/// ----------------------------------------------------------------------------------------------------

protected virtual void EvaluateSeparator(char separator)
{
if (!(string.IsNullOrEmpty(this.nameB)) && (this.nameB.Contains(separator))) {
throw new ArgumentException(message: "The parameter separator cannot be any character contained in the parameter name.", paramName: "separator");

} else {
this.separatorB = separator;

}

}

#endregion

}

#endregion

#region " Commandline Parameter (Of Object) "

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Represents a Commandline Parameter.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
public sealed class CommandlineParameter : CommandlineParameter<object>
{}

#endregion

}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================





Una vez tenemos la base de como representar de forma abstracta un parámetro commandline, creamos las instancias de nuestro Type para crear los parámetros.

( En este ejemplo creo un parámetro que toma como valor una cadena de texto, otro parámetro que toma como valor un booleano, y un array con dichos parámetros. )

Código (csharp) [Seleccionar]
private readonly CommandlineParameter<string> param1 = new CommandlineParameter<string> {
Name = "/Switch1",
Separator = '=',
DefaultValue = "Hello World",
IsRequired = true
};

private readonly CommandlineParameter<bool> param2 = new CommandlineParameter<bool> {
Name = "/Switch2",
Separator = '=',
DefaultValue = false,
IsRequired = false
};

CommandlineParameter[] @params = {
param1,
param2
};


El modo en que se usuaría la aplicación sería el siguiente:

Aplicación.exe /switch1="valor1" /switch2="true" o "false"
(el orden de los parámetros no importa, y las comillas dobles tampoco, excepto para encerrar strings con espacios en blanco)




Ahora nos faltaría desarrollar el algoritmo de un método que tome como argumento (pasando por referencia) la representación de los parámetros para asignarles los valores y (pasando por valor) los argumentos commandline de la aplicación.

Nota: Adicionalmente le añadí dos parámetros que toman delegados para notificar cuando se encuentra un error de sintaxis o un parámetro requerido que no ha sido encontrado en los argumentos.

Código (csharp) [Seleccionar]
/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Loop through all the command-line arguments of this application.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <param name="params">
/// The commandline parameters.
/// </param>
///
/// <param name="callbackSyntaxError">
/// An encapsulated method that is invoked if a syntax error is found in one of the arguments.
/// </param>
///
/// <param name="callbackMissingRequired">
/// An encapsulated method that is invoked if a required parameter is missing in the arguments.
/// </param>
/// ----------------------------------------------------------------------------------------------------
private void ParseArguments(ref CommandlineParameter[] @params, Action<CommandlineParameter> callbackSyntaxError, Action<CommandlineParameter> callbackMissingRequired) {
ParseArguments(ref @params, Environment.GetCommandLineArgs.Skip(1), callbackSyntaxError, callbackMissingRequired);
}

/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Loop through all the command-line arguments of this application.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <param name="params">
/// The commandline parameters.
/// </param>
///
/// <param name="args">
/// The collection of commandline arguments to examine.
/// </param>
///
/// <param name="callbackSyntaxError">
/// An encapsulated method that is invoked if a syntax error is found in one of the arguments.
/// </param>
///
/// <param name="callbackMissingRequired">
/// An encapsulated method that is invoked if a required parameter is missing in the arguments.
/// </param>
/// ----------------------------------------------------------------------------------------------------
private void ParseArguments(ref CommandlineParameter[] @params, IEnumerable<string> args, Action<CommandlineParameter> callbackSyntaxError, Action<CommandlineParameter> callbackMissingRequired) {

List<CommandlineParameter> paramRequired = (from param in @paramswhere param.IsRequired).ToList;

foreach (string arg in args) {

foreach (CommandlineParameter param in @params) {

if (arg.StartsWith(param.Name, StringComparison.OrdinalIgnoreCase)) {
if (!arg.Contains(param.Separator)) {
callbackSyntaxError.Invoke(param);
return;

} else {
string value = arg.Substring(arg.IndexOf(param.Separator) + 1);

if ((paramRequired.Contains(param))) {
paramRequired.Remove(param);
}

if (string.IsNullOrEmpty(value)) {
param.Value = param.DefaultValue;
continue;

} else {
try {
param.Value = Convert.ChangeType(value, param.DefaultValue.GetType());
continue;

} catch (Exception ex) {
callbackSyntaxError.Invoke(param);
return;

}

}

}

}

}

}

if ((paramRequired.Any)) {
callbackMissingRequired.Invoke(paramRequired.First);
}

}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================





Por último, así es como usariamos dicho método:

Código (csharp) [Seleccionar]
static void Main(string[] args) {
   ParseArguments(ref @params, args, Parameter_OnSyntaxError, Parameter_OnMissingRequired)
   // Llegados a este punto todo fue exitoso. Continuar con la lógica...
}

private void Parameter_OnSyntaxError(CommandlineParameter param) {
Console.WriteLine(string.Format("[X] Syntax error in parameter: {0}", param.Name));
Environment.Exit(exitCode: 1);
}

private void Parameter_OnMissingRequired(CommandlineParameter param) {
Console.WriteLine(string.Format("[X] Parameter required: {0}", param.Name));
Environment.Exit(exitCode: 1);
}


Nota: La declaración del array params está en el código de arriba.

Una imagen:





No se si este tocho que he escrito será demasiado tedioso (demasiado código para vagos xD) pero ahí lo dejo para quien le sirva la idea.

Saludos!








HCK.

Hola Elektro, estos dias no he podido estar mucho por aquí, pero menuda currada, me has hecho todo el trabajo prácticamente. De todas maneras busqué la información para aprender más, y de verdad que en general te lo agradezco mucho.  ;D

A pesar de haber empezado hace un tiempo con este lenguaje y la plataforma .NET en general, siempre voy aprendiendo a tramos por la falta de tiempo, pero me interesa mucho y por eso lo saco adelante, y con personas como tu las cosas son mas fáciles.  :)

Ya que estamos, y para no molestarte mucho mas con el asunto... ¿Algún libro que me pudieses recomendar sobre C#, a ser posible en Español (he leido que los que hay son penosos...) o en inglés pero que valga la pena pagar por él?

Gracias, si necesitas algo aquí tienes un compañero.

Un saludo!