12.30.2010

IComparer<T>, IComparable<T> y el SortedList<TKey, TValue>

Revisando la implementación del SortedList y como alterar los criterios de comparación me di a la tarea de buscar alternativas para resolver un problema que tenía en un software en el que estaba trabajando.

Resulta que cuando uno utiliza un SortedList tiene dos opciones para manejar el ordenamiento de la misma: IComparable e IComparer. ¿Y cuál es la diferencia? Bueno desde el punto de vista desarrollo de aplicaciones 2:

  1. Tengo acceso al código de la clase que quiero ordenar.
  2. La clase esta en un componente a y no tengo acceso a su código.

IComparable<T>

En primera instancia, supongamos que tenemos acceso al código de la clase que queremos tener dentro de nuestro SortedList. En este caso utilizamos la interface IComparable<T> ya que el SortedList utiliza el método CompareTo de la implementación de IComparable<T> para realizar la ordenación; esta interface además es la que le da el servicio para ordenar al método Sort del Array. Para poder utilizar esta interface debemos implementarla en la clase que vamos a utilizar tal y como lo muestra el siguiente código.

public class Estudiante : IComparable<Estudiante>
{
public int Id { get; set; }
public string Nombre { get; set; }
public string Apellidos { get; set; }

public Estudiante( int pId, string pNombre, string pApellidos )
{
Id = pId;
Nombre = pNombre;
Apellidos = pApellidos;
}

public int CompareTo( Estudiante other )
{
return this.Apellidos.CompareTo( other.Apellidos );
}

}

Con esta implementación estamos haciendo que cuando se use la clase Estudiante en colecciones genéricas se ordene por los apellidos del estudiante. Si vemos la siguiente lista y la recorremos, veremos que los ítems fueron ordenados por medio de los apellidos.
SortedList<Estudiante, Estudiante> _listaEstudiantes = new SortedList<Estudiante, Estudiante>();

_listaEstudiantes.Add( _estudiante1, _estudiante1 );
_listaEstudiantes.Add( _estudiante2, _estudiante2 );
_listaEstudiantes.Add( _estudiante3, _estudiante3 );
_listaEstudiantes.Add( _estudiante4, _estudiante4 );
_listaEstudiantes.Add( _estudiante5, _estudiante5 );
_listaEstudiantes.Add( _estudiante6, _estudiante6 );
_listaEstudiantes.Add( _estudiante7, _estudiante7 );

foreach (KeyValuePair<Estudiante, Estudiante> item in _listaEstudiantes)
{
Console.WriteLine( "Apellidos: {0} | Nombre: {1}", item.Value.Apellidos, item.Value.Nombre );
}
Al ejecutar este código el resultado será el siguiente:
image

IComparer<T>


Entonces cual es el uso que le vamos a dar a IComparer<T>. Bueno, en realidad lo que nos da es un mecanismo extra de comparación para ordenar la lista. Este mecanismo es útil cuando por ejemplo queremos utilizar una lista ordenada para la clase pero no tenemos el código de la clase y por lo tanto no podemos implementar la interface IComparable<T>. Supongamos la siguiente clase Estudiante:
public class Estudiante 
{
public int Id { get; set; }
public string Nombre { get; set; }
public string Apellidos { get; set; }

public Estudiante( int pId, string pNombre, string pApellidos )
{
Id = pId;
Nombre = pNombre;
Apellidos = pApellidos;
}

}

Supongamos que viene en un dll que no podemos modificar y sin embago la queremos ordenar por el criterio que nosotros deseamos. En este caso podemos utilizar la interface IComparer<T>. Lo que tenemos que hacer es crear una clase que implemente la interface IComparer<T> y establecer ahi la forma en que queremos hacer la comparación, en este caso por apellidos.

public class ComparadorEstudiante : IComparer<Estudiante>
{
public int Compare( Estudiante x, Estudiante y )
{
return x.Apellidos.CompareTo(y.Apellidos);
}
}
Seguidamente procedemos a crear la instancia de la clase SortedList<T> indicándole que utilice este comparador en el constructor.

SortedList<Estudiante, Estudiante> _estudiantes = new SortedList<Estudiante, Estudiante>( new ComparadorEstudiante() );

Estudiante _estudiante1 = new Estudiante( 1, "Ana", "Hernandez" );
Estudiante _estudiante2 = new Estudiante( 2, "Luis", "Ruiz" );
Estudiante _estudiante3 = new Estudiante( 3, "Esther", "Benavidez" );
Estudiante _estudiante4 = new Estudiante( 4, "Ulises", "Alonso" );
Estudiante _estudiante5 = new Estudiante( 5, "Ana", "Lopez" );
Estudiante _estudiante6 = new Estudiante( 6, "Mario", "Estrada" );
Estudiante _estudiante7 = new Estudiante( 7, "Clara", "Rodriguez" );

_estudiantes.Add( _estudiante1, _estudiante1 );
_estudiantes.Add( _estudiante2, _estudiante2 );
_estudiantes.Add( _estudiante3, _estudiante3 );
_estudiantes.Add( _estudiante4, _estudiante4 );
_estudiantes.Add( _estudiante5, _estudiante5 );
_estudiantes.Add( _estudiante6, _estudiante6 );
_estudiantes.Add( _estudiante7, _estudiante7 );

foreach (KeyValuePair<Estudiante, Estudiante> item in _estudiantes)
{
Console.WriteLine( "Apellidos: {0} | Nombre: {1}", item.Value.Apellidos, item.Value.Nombre );
}
Ahora la comparación se hará a través de la clase ComparadorEstudiante y el resultado será el siguiente:
image


Algunos podrán decir que para que tanta complicación si se podría pasar en el TKey del KeyValuePair el campo por el que quiero la llave y a partir de ahi que se genere la ordenación por el comparador por defecto del mismo, sin embargo; en el caso donde uno requiere este tipo de ordenación es principalmente cuando la clase que queremos ordenar tiene una relación de 1 a 1 con una clase X y es la que nos va a permtir hacer esta relación; un ejemplo típico es una clase proyecto asociada con una clase ProjectManager y queremos los proyectos ordenados por el project manager.


12.03.2010

Taller de Cloud Computing y Windows Azure

 

Este post no es técnico, más bien es una invitación a un taller que se va a llevar a cabo en el mes de enero acerca de Cloud Computing y Windows Azure y que tendré la dicha de dictar. Espero nos puedan acompañar. Mas información a los teléfonos o direcciones en el brochure o a mi correo diego.rojas [at] formsevolution.net. Hay promos disponibles.

Seminario Cloud Comp

Technorati Tags: ,,

12.02.2010

Utilizando Unity para Resolver un Problema de Negocio - 4

Continuando con la serie de posts acerca de como utilizar Unity y aplicarlo a un problema de negocio, vamos a agregar en este post la funcionalidad necesaria para permitir agregar nuevos tipos de usuario sin tener que compilar nuestra capa de negocios. La idea con este post es permitirle al desarrollador crear componentes que puedan ser asociados a la lógica de negocios sin tener que modificar la misma a través de su código, y que simplemente con agregar el nuevo tipo al archivo de configuración de nuestra aplicación, este pueda utilizarlo. Para ejemplificar mejor el problema voy a mostrar un diagrama de como se vería físicamente nuestro programa. El estado actual de la aplicación a nivel de lógica de negocios es el siguiente:

image

La idea de este post es crear un nuevo usuario que se va a llamar usuario federado, el cual no va a formar parte de nuestro proyecto de Core.LogicaDeNegocios, si no que será un dll aparte el cual implementa una clase del tipo IUsuario llamada Usuario_Federado. Nuestro esquema debería verse de la siguiente manera:

image

Como podemos ver en la figura anterior, ahora vamos a agregar un nuevo dll en donde va a residir nuestro nuevo usuario – en color rojo. Lo interesante con esta solución es que vamos a poder agregar en nuestro BL nuevos usuarios sin tener que ir a recompilar ni modificar nuestra lógica de negocios; esto es lo que vamos a hacer en el post del día de hoy – para que el cambio se pueda manejar de la misma manera en la capa de UI vamos a utilizar Prism en post posteriores.

Solución

En primera instancia vamos a crear un nuevo proyecto del tipo librería y le vamos a poner Core.LogicaDeNegocios.Usuario_Federado. Seguidamente procedemos a crear la clase Usuario_Federado la cual es similar a las demas clases que implementan IUsuario definidas dentro de la librería Core.LogicaDeNegocios.dll. El dll que acabamos de crear requiere una referencia al dll donde esta definido el tipo IUsuario, es decir Core.LogicaDeNegocios.dll. El código de esta nueva clase es el siguiente:

using Core.LogicaDeNegocios;

using System.Diagnostics;

namespace Core.LogicaDeNegocios.Usuario_Federado
{
public class Usuario_Federado : IUsuario
{
IEmpresa _empresa;
public IEmpresa Empresa
{
get
{
return _empresa;
}
set
{
_empresa = value;
}
}

public IUsuario Autenticar()
{
Trace.WriteLine("Autenticar Usuario_Federado");
return this;
}

public void Autorizar(object _form)
{
Trace.WriteLine("Autorizar Usuario Federado");
}
}
}


Ahora vamos a modificar el UI para que se puedan crear instancias de la clase residente en este nuevo dll. Agregamos la “empresa D” la cual estará asociada con el Usuario_Federado y luego procedemos a agregar el siguiente código en el botón con el cual estamos cargando el usuario de acuerdo a los datos ingresados.



string _tipoUsuario = string.Empty;
if (cmbEmpresa.SelectedItem.ToString() == "Empresa A")
_tipoUsuario = "usuario";
else if (cmbEmpresa.SelectedItem.ToString() == "Empresa B")
_tipoUsuario = "usuarioWeb";
else if (cmbEmpresa.SelectedItem.ToString() == "Empresa C")
_tipoUsuario = "otroUsuario";
else
_tipoUsuario = "usuarioFederado";
UsuarioActual = AdministradorUsuario.ObtenerInstancia(_tipoUsuario, txtNombreUsuario.Text, txtPassword.Text).Autenticar();



El paso siguiente es modificar el App.Config para mapear el tipo, es decir IUsuario a Usuario_Federado:



      <register type="IUsuario" name="usuarioFederado" mapTo="Usuario_Federado" />


Por último, ejecutamos la aplicación y podemos ver que si seleccionamos la empresa D, obtenemos una instancia del tipo Usuario_Federado.



image



Como se puede ver en la figura anterior tenemos una instancia de Usuario_Federado, yal y como lo esperabamos. Nótese sin embargo que la empresa esta en nulo pese a que tenemos implementada la propiedad Empresa del tipo IEmpresa ¿Qué nos falta? Pues si vamos a utilizar las empresas ya definidas en nuestro dll de Core.LogicaDeNegocios.dll simplemente tenemos que agregar el mapeo de la propiedad y listo.



 <register type="IUsuario" name="usuarioFederado" mapTo="Usuario_Federado" >
<
property name="Empresa" dependencyName ="empresaMap" dependencyType="IEmpresa"></property>
</
register>



El resultado será el siguiente:



image



Es importante darnos cuenta que hemos agregado una nueva entidad de negocios a nuestra lógica de negocios sin necesidad de recompilar la misma, esto nos dará muchas ventajas cuando queramos crear aplicaciones listas para evolucionar en el futuro. En el próximo post vamos a analizar los beneficios de este tipo de arquitecturas y cuales frameworks y aplicaciones utilizan este tipo de diseño.



Technorati Tags: ,,,,

11.29.2010

Utilizando Unity para Resolver un Problema de Negocio - 3

Siguiendo con el uso de Unity para trabajar problemas de negocios de una forma menos dependiente entre los componentes que desarrollamos, vamos ahora a tratar el caso en donde la clase recién instanciada tiene una referencia a otra clase que a su vez debe ser resuelta por el contenedor de IoC.

Supongamos que cada usuario tiene una relación de agregación con la interface IEmpresa y que dependiendo de la empresa se paga un cargo diferente cuando se realiza alguna transacción. La interface IEmpresa tendrá un método que solamente le aplica el cargo a la transacción. La relación de clases en el problema actual es la siguiente:

image

En nuestro código la implementación de nuestras clases lucirá así:

Interface IEmpresa

namespace Core.LogicaDeNegocios
{
public interface IEmpresa
{
void AplicarCargos();
}
}



Clase Oficina: Implementa IEmpresa



using System.Diagnostics;

namespace Core.LogicaDeNegocios
{
public class Oficina: IEmpresa
{
public void AplicarCargos()
{
Trace.WriteLine("Aplicar Cargos Oficina");
}
}
}


Clase EmpresaExterna: Implementa IEmpresa



using System.Diagnostics;

namespace Core.LogicaDeNegocios
{
public class EmpresaExterna: IEmpresa
{
public void AplicarCargos()
{
Trace.WriteLine("Aplicar Cargos Empresa Externa");
}
}
}



En la interface IUsuario agregamos la relación de agregación antes representada:



public interface IUsuario
{
IEmpresa Empresa { get; set; }

IUsuario Autenticar();
void Autorizar(object _form);
}



Ahora modificamos la clase Usuario, para que tenga una implementación de IEmpresa. Nótese que la clase debe implementar la propiedad expuesta en la interface y no necesariamente el mismo tipo en el atributo, ya que con solo tener una clase que implemente la interface, ya podemos crear la asociación.



public class Usuario : IUsuario
{
Oficina _oficina;

public Usuario(string pNombreUsuario, string pPassword)
{
Trace.WriteLine(string.Format("Datos del usuario. NombreUsuario: {0}, Password: {1}", pNombreUsuario, pPassword));
}

public IUsuario Autenticar()
{
Trace.WriteLine("Autenticar Usuario");
return this;
}

public void Autorizar(object _form)
{
Trace.WriteLine("Autenticar Usuario");
}

public IEmpresa Empresa
{
get
{
return _oficina;
}
set
{
_oficina = value as Oficina;
}
}

}



Lo mismo ocurrirá con la clase Usuario_Web, con la diferencia que en Usuario_Web si vamos a crear un atributo del tipo IEmpresa.



public class Usuario_Web: IUsuario
{
IEmpresa _empresa;

public Usuario_Web(string pNombreUsuario, string pPassword)
{
Trace.WriteLine(string.Format("Datos del usuario. NombreUsuario: {0}, Password: {1}", pNombreUsuario, pPassword));
}

public virtual IUsuario Autenticar()
{
Trace.WriteLine("Autenticar Usuario Web");
return this;
}

public void Autorizar(object _form)
{
Trace.WriteLine("Autorizar Usuario Web");
}

public IEmpresa Empresa
{
get
{
return _empresa;
}
set
{
_empresa = value;
}
}
}


Por otro lado, vamos a crear una clase que hereda de Usuario_Web y que implementa su propia implementación de IEmpresa.



public class OtroUsuario : Usuario_Web
{
IEmpresa _empresa;

public OtroUsuario(string pNombreUsuario, string pPassword)
:base(pNombreUsuario, pPassword)
{
Trace.WriteLine(string.Format("Datos del usuario. NombreUsuario: {0}, Password: {1}", pNombreUsuario, pPassword));
}

public override IUsuario Autenticar()
{
Trace.WriteLine("Autenticar Otro Usuario Web");
return this;
}

public IEmpresa Empresa
{
get
{
return _empresa;
}
set
{
_empresa = value;
}
}

}



Por último agregamos el nuevo mapping para tenerlo entre la lista de clases que pueden ser resueltas.



  <unity xmlns="http://schemas.microsoft.com/practices/2010/unity">
<
alias alias="IUsuario" type="Core.LogicaDeNegocios.IUsuario, Core.LogicaDeNegocios" />
<
namespace name="Core.LogicaDeNegocios" />
<
assembly name="Core.LogicaDeNegocios" />
<
container>
<
register type="IUsuario" name="usuario" mapTo="Usuario" />
<
register type="IUsuario" name="usuarioWeb" mapTo="Usuario_Web" />
<
register type="IUsuario" name="otroUsuario" mapTo="OtroUsuario" />
</
container>
</
unity>



En la parte del UI, agregamos el nuevo tipo de usuario a través de la empresa C.



private void btnLogin_Click(object sender, EventArgs e)
{
string _tipoUsuario = string.Empty;
if (cmbEmpresa.SelectedItem.ToString() == "Empresa A")
_tipoUsuario = "usuario";
else if (cmbEmpresa.SelectedItem.ToString() == "Empresa B")
_tipoUsuario = "usuarioWeb";
else
_tipoUsuario = "otroUsuario";

UsuarioActual = AdministradorUsuario.ObtenerInstancia(_tipoUsuario, txtNombreUsuario.Text, txtPassword.Text).Autenticar();

}



Si ejecutamos el código tal y como esta – y llevando la continuación del post anterior – veremos lo siguiente:



image



Como vemos en la figura, la instancia de empresa del tipo que resolvió Unity esta en nulo. Sin duda no es el comportamiento que queremos. Pero como hacer para que la clase que estamos instanciando pueda resolver la empresa que desea en tiempo de ejecución? Para este problema en específico, Unity me permite crear inyección para las propiedades con lo cual puedo definir dentro del registro del tipo para mapear, como debe de cargarse una propiedad de la clase que se va a resolver. Para lograr esto vamos a modificar nuestro archivo de configuración de la siguiente forma:



  <unity xmlns="http://schemas.microsoft.com/practices/2010/unity">
<
alias alias="IUsuario" type="Core.LogicaDeNegocios.IUsuario, Core.LogicaDeNegocios" />
<
namespace name="Core.LogicaDeNegocios" />
<
assembly name="Core.LogicaDeNegocios" />
<
container>
<
register type="IEmpresa" name="empresaMap" mapTo="EmpresaExterna" />
<
register type="IEmpresa" name="oficinaMap" mapTo="Oficina"
/>
<
register type="IUsuario" name="usuario" mapTo="Usuario" >
<
property name="Empresa" dependencyName ="oficinaMap" dependencyType="IEmpresa"></property
>
</
register
>
<
register type="IUsuario" name="usuarioWeb" mapTo="Usuario_Web" />
<
register type="IUsuario" name="otroUsuario" mapTo="OtroUsuario" >
<
property name="Empresa" dependencyName ="empresaMap" dependencyType="IEmpresa"></property
>
</
register
>
</
container>
</
unity>



Como podemos ver en el texto en negrita, primera vamos a crear dos mapas nuevos para los tipos recién creados que implementan la interface IEmpresa.



<register type="IEmpresa" name="empresaMap" mapTo="EmpresaExterna" />
<
register type="IEmpresa" name="oficinaMap" mapTo="Oficina" />



Seguidamente vamos a modificar los registros para los usuarios deseados y vamos a establecer la instancia que queremos sea inyectada en la propiedad del tipo de IEmpresa.



<register type="IUsuario" name="otroUsuario" mapTo="OtroUsuario" >
<
property name="Empresa" dependencyName ="empresaMap" dependencyType="IEmpresa"></property>
</
register>



      <register type="IUsuario" name="usuario" mapTo="Usuario" >
<
property name="Empresa" dependencyName ="oficinaMap" dependencyType="IEmpresa"></property>
</
register>





En este caso hemos establecido que cuando se vaya a crear una instancia del tipo Usuario la propiedad IEmpresa se instancie con el tipo Oficina, y si por lo contrario creamos una instacia del tipo OtroUsuario vamos a tener la propiedad IEmpresa con una instancia del tipo EmpresaExterna. Si creamos una instancia de Usuario_Web la instancia de la propiedad IEmpresa no se podrá resolver y por lo tanto será null.



Si seleccionamos Empresa C –> OtroUsuario el resultado será:



image



Si seleccionamos Empresa A –> Usuario el resultado será:



image



Y si seleccionamos Empresa B –> el resultado será:



image







Technorati Tags: ,,

11.23.2010

Utilizando Unity para Resolver un Problema de Negocio - 2

En el post anterior presentamos un problema el cual resolvimos usando Unity, sin embargo, este problema se presentó de una manera muy simple y un poco corto en realidad. En este post vamos a agregarle un par de situaciones que se presentan normalmente en este tipo de sistemas.

En primer lugar, cuando creamos la instancia del usuario que necesitamos, le tuvimos que indicar de forma explícita al método AdministradorDeUsuario.ObtenerInstancia utilizando un generic, algo que por supuesto no es deseado ya que obliga al cliente de la aplicación a estar conciente de los tipos de usuario que existen y esto limita la aplicación a trabajar con los usuarios existentes y no nos permite agregar nuevos usuarios sin tener que recompilar la aplicación.

En segundo lugar estamos utilizando clases para los usuarios que no contienen datos y por lo tanto no son inicializados cuando se crea la instancia, por lo que el método Autorizar no funcionaría en un escenario real ya que no tiene los datos que el usuario utilizó para loguearse.

Mejoras

Para atacar el primer problema vamos a cambiar la forma en que le solicitamos nuestras instancias a Unity. Con los contenedores de IoC tenemos la gran ventaja de que podemos solicitarles que resuelvan el tipo en base al nombre del mapeo que queremos utilizar. El primer paso es cambiar el archivo de configuración para que trabaje con un solo contenedor de la siguiente forma:

<?xml version="1.0" encoding="utf-8" ?>
<
configuration>
<
configSections>
<
section name="unity" type="Microsoft.Practices.Unity.Configuration.UnityConfigurationSection, Microsoft.Practices.Unity.Configuration" />
</
configSections>
<
unity xmlns="http://schemas.microsoft.com/practices/2010/unity">
<
alias alias="IUsuario" type="Core.LogicaDeNegocios.IUsuario, Core.LogicaDeNegocios" />
<
namespace name="Core.LogicaDeNegocios" />
<
assembly name="Core.LogicaDeNegocios" />
<
container>
<
register type="IUsuario" name="usuario" mapTo="Usuario"
/>
<
register type="IUsuario" name="usuarioWeb" mapTo="Usuario_Web"
/>
</
container
>
</
unity>
</
configuration>



Seguidamente quitamos el parámetro del tipo que estamos solicitando en el método ObtenerInstancia ya que como dije anteriormente vamos a dejar que Unity resuelva el tipo en base al nombre del mapa que queremos utilizar. Los posibles nombres para los mapas son:



<register type="IUsuario" name="usuario" mapTo="Usuario" />
<
register type="IUsuario" name="usuarioWeb" mapTo="Usuario_Web" />



Además al método ObtenerInstancia le vamos a agregar dos parámetros más, el usuario y el password para que pueda agregarlos como valores al usuario que se vaya a crear – problema 2. La invocación del método desde nuestra pantalla de login ahora se ve así:



string _tipoUsuario = string.Empty;
if (cmbEmpresa.SelectedItem.ToString() == "Empresa A")
_tipoUsuario = "usuario";
else
_tipoUsuario = "usuarioWeb";

UsuarioActual = AdministradorUsuario.ObtenerInstancia(_tipoUsuario, txtNombreUsuario.Text, txtPassword.Text).Autenticar();



El siguiente paso y para trabajar en el problema 2, es crear un constructor para cada tipo de usuario de forma tal que reciba el usuario y el password como parámetros y que pueda hacer la autenticación con los mismos. Las clases modificadas son las siguientes:



public class Usuario_Web: IUsuario
{

public Usuario_Web(string pNombreUsuario, string pPassword)
{
Trace.WriteLine(string.Format("Datos del usuario. NombreUsuario: {0}, Password: {1}", pNombreUsuario, pPassword));
}


public IUsuario Autenticar()
{
Trace.WriteLine("Autenticar Usuario Web");
return this;
}

public void Autorizar(object _form)
{
Trace.WriteLine("Autorizar Usuario Web");
}
}



public class Usuario : IUsuario
{
public Usuario(string pNombreUsuario, string pPassword)
{
Trace.WriteLine(string.Format("Datos del usuario. NombreUsuario: {0}, Password: {1}", pNombreUsuario, pPassword));
}


public IUsuario Autenticar()
{
Trace.WriteLine("Autenticar Usuario");
return this;
}

public void Autorizar(object _form)
{
Trace.WriteLine("Autenticar Usuario");
}
}



Ahora solo nos queda modificar el método ObtenerInstancia para que a través de Unity se pueda resolver el tipo que deseamos instanciar en base al tipo de usuario que queremos utilizar y además que los datos del usuario puedan ser inyectados a través del constructor de la clase a instanciar:



using Microsoft.Practices.Unity;
using Microsoft.Practices.Unity.Configuration;

namespace Core.LogicaDeNegocios
{
public static class AdministradorUsuario
{
public static IUsuario ObtenerInstancia(string mapName, string pUsuario, string pPassword)
{
using (IUnityContainer container = new UnityContainer())
{
container.LoadConfiguration();
return container.Resolve<IUsuario>(mapName, new ParameterOverrides { { "pNombreUsuario", pUsuario }, { "pPassword", pPassword } });

}
}
}
}



Con esto ya podemos ejecutar nuestra aplicación y crear la instancia que deseamos con los datos enviados.



image



image



Technorati Tags: ,,,,

11.22.2010

Utilizando Unity para Resolver un Problema de Negocio - 1

En días pasados me encontré con un problema en el cual se podría utilizar IoC para resolverlo de forma sencilla y elegante. En este post voy a reproducir este fix con un problema un tanto diferente pero en el cual se puede ver como de forma sencilla podemos aprovecharnos de las bondades de Unity y a la vez hacer nuestra aplicación más versátil y fácil de hacer crecer.

El Problema

Supongamos que tenemos un sistema que hace una seria de funciones para la empresa y que luego de una fusión con otra empresa, tenemos un sistema similar pero que además realiza unas funciones extras. Resulta que cada sistema tiene su propia tabla de usuarios ( así es, como siempre primero se hacen los sistemas y luego se piensa en la seguridad y no a la inversa como debería ser ) y se quiere hacer una versión de forma rápida que mezcle las funcionalidades de ambas aplicaciones. La aplicación va a tener un ComboBox en donde el usuario indica a cual empresa originalmente pertenece y el sistema debe resolver a donde debe de ir a autenticarse el usuario. En primera instancia, la primera aplicación es desktop y la segunda aplicación es Web. El problema radica en cómo manejar estos tipos diversos de usuario en una sola aplicación sin necesidad de andar previendo con cual tipo estamos trabajando.

La Solución

La solución a este problema esta por el lado del polimorfismo, ya que si definimos una interface IUsuario, podemos trabajar con la interface y no con el tipo concreto, lo cual nos ahorraría el peso de estar manejando un tipo de usuario X en la aplicación cuando se requiera. El siguiente diagrama nos muestra lo expuesto anteriormente.

image

El código de cada una de las clases es el que se presenta a continuación:

namespace Core.LogicaDeNegocios
{
//Este usuario se autentica en una base de datos para usuarios Web
public class Usuario_Web: IUsuario
{
public IUsuario Autenticar()
{
Trace.WriteLine("Autenticar Usuario Web");
return this;
}

public void Autorizar(object _form)
{
Trace.WriteLine("Autorizar Usuario Web");
}
}
}



namespace Core.LogicaDeNegocios
{
//Este usuario se autentica utilizando una base de datos de usuarios locales
public class Usuario : IUsuario
{
public IUsuario Autenticar()
{
Trace.WriteLine("Autenticar Usuario");
return this;
}

public void Autorizar(object _form)
{
Trace.WriteLine("Autenticar Usuario");
}
}
}



Ahora el punto es como hacer que la aplicación construya la instancia que corresponda – dependiendo de la selección del comboBox – sin necesidad de estar creando clases concretas por toda la aplicación. Para este caso es que se decidió utilizar Unity. Con Unity tenemos la posibilidad de registrar los tipos que deseamos y mapearlos con clases concretas para cuando solicitemos el tipo, se nos retorne la instancia que deseamos. Para registrar nuestras clases anteriores vamos a utilizar el archivo app.config de nuestra aplicación desktop y vamos a agregar la siguiente configuración.



<?xml version="1.0" encoding="utf-8" ?>
<
configuration>
<
configSections>
<
section name="unity" type="Microsoft.Practices.Unity.Configuration.UnityConfigurationSection, Microsoft.Practices.Unity.Configuration" />
</
configSections>
<
unity xmlns="http://schemas.microsoft.com/practices/2010/unity">
<
alias alias="IUsuario" type="Core.LogicaDeNegocios.IUsuario, Core.LogicaDeNegocios" />
<
namespace name="Core.LogicaDeNegocios" />
<
assembly name="Core.LogicaDeNegocios" />
<
container>
<
register type="IUsuario" name="usuario" mapTo="Usuario"
/>
</
container
>
<
container name="UsuarioWeb"
>
<
register type="IUsuario" name="usuarioWeb" mapTo="Usuario_Web"
/>
</
container
>
</
unity>
</
configuration>



Como podemos ver en el xml anterior, estamamos registrando dos tipos con la interface IUsuario: Usuario y Usuario_Web, para que cuando solicitemos el tipo Unity tenga como resolver y pueda crear la instancia que deseamos. El siguiente paso es utilizar el contenedor de Unity para que nos resuelva el tipo y dado que no queremos que el UI se tenga que preocupar por este block, vamos a crear una clase estática que nos ayude retornando la instancia a partir de un tipo especificado como parámetro en una función que vamos a crear para crear la instancia de acuerdo al tipo que solicita el usuario.



//Unity
using Microsoft.Practices.Unity;

namespace Core.LogicaDeNegocios
{
public static class AdministradorUsuario
{
public static IUsuario ObtenerInstancia<T>() where T:IUsuario
{
using (IUnityContainer container = new UnityContainer())
{
return container.Resolve<T>();
}
}
}
}


Nótese el constraint en el Generic, el cual nos delimita los tipos de datos que pueden ser T, en este caso T tiene que implementar IUsuario. Por último procedemos a crear el código en la forma de log in, en donde de acuerdo a la opción seleccionada por el usuario vamos a seleccionar el tipo que queremos pasar como T a la función ObtenerInstancia<T>.



IUsuario _usuarioActual;

public IUsuario UsuarioActual
{
get { return _usuarioActual; }
set { _usuarioActual = value; }
}

private void btnLogin_Click(object sender, EventArgs e)
{
if (cmbEmpresa.SelectedItem == "Empresa A")
UsuarioActual = AdministradorUsuario.ObtenerInstancia<Usuario>().Autenticar();
else
UsuarioActual = AdministradorUsuario.ObtenerInstancia<Usuario_Web>().Autenticar();

}



En este caso específico, estamos pasando el tipo concreto al método que nos devuelve una instancia de la clase solicitada, y como ambas clases deben de implementar la interface IUsuario podemos invocar el método Autenticar que es parte de este contrato. Este método nos devuelve una instancia de IUsuario el cual podemos utilizar en nuestra aplicación para operaciones posteriores de Autorización. Para cerrar el post vemos la estructura de proyecto utilizada, la cual como veremos en el siguiente post nos va a permitir cargar clases de lógica de  negocios dependiendo de la forma en que se configure el sistema.



image



Technorati Tags: ,,

11.20.2010

Actualizar una Entidad Utilizando el ObjectDataSource y el EntityFramework

Continuando con el post respecto a utilizar el EntityFramework en conjunto con el ObjectDataSource en una aplicación ASP.NET, vamos a ver como actualizar una entidad utilizando esta combinación de componentes. En primera instancia vamos a tener una tabla de usuarios la cual una vez creado el modelo en el EntityFramework vamos a tener una estructura como la que se ve en la siguiente figura.

image

Luego procedemos a crear nuestra lógica de negocio en donde vamos a tener un método que nos permite obtener una instancia única del contexto de las entidades del modelo del EntityFramework, este método lo podemos ver a continuación.

Public Class SeguridadBL
Public Shared Contexto As SeguridadEntities

Public Shared Function ObtenerContexto() As SeguridadEntities
If Contexto Is Nothing Then
Contexto = New SeguridadEntities
End If
Return
Contexto
End Function


El siguiente método es el que nos sirve para actualizar la entidad. Como podemos ver en el método, vamos a recibir una instancia de una variable tipo Usuario, la cual vamos a tener que trabajar un poco más para poder actualizar los datos en la base de datos.



    Public Function ActualizarUsuario(ByVal pUsuario As Usuario) As Integer

Dim
resultado = From u In ObtenerContexto.Usuarios _
Where u.Id = pUsuario.Id _
Select u
Dim tmpUsuario As Usuario = resultado.SingleOrDefault
tmpUsuario.Nombre = pUsuario.Nombre
tmpUsuario.Apellido1 = pUsuario.Apellido1
tmpUsuario.Apellido2 = pUsuario.Apellido2
tmpUsuario.Activo = pUsuario.Activo
tmpUsuario.Email = pUsuario.Email
tmpUsuario.Password = pUsuario.Password
Return ObtenerContexto.SaveChanges()
End Function



Como la instancia de la clase usuario que viene en el parámetro no pertence al contexto actual que estamos utilizando, vamos a tener agregarla al contexto. Sin embargo, para poder agregar la instancia, tendríamos que agregarle el entityKey, el cuál de por si ya existe en el contexto. En este caso, vamos a ir a traer la instancia utilizando una simple operación de select y luego, le vamos a setear los valores desde la instancia que viene por parámetro – en mi opinión una solución no muy elegante, pero al parecer la más practica. Pot último, le decimos al contexto que guarde los cambios en la base de datos con SaveChanges.



Del lado de la página ASP.NET, vamos a crear un objeto del tipo ObjectDataSource y vamos a proceder a configurarlo de la siguiente manera:



1. Primero seleccionamos la clase desde la cual vamos a tener el método de actualización, en nuestro caso SeguridadBL.



image



2. Luego procedemos a configurar el método de actualización. Cuando el objectDataSource detecta que el parámetro es un tipo complejo, establece la propiedad DataObjectTypeName con el nombre del tipo del parámetro el cual utiliza para crear una instancia de la clase cuando es invocado el método.



image



3. Seguidamente le asociamos el ObjectDataSource al GridView.



image



4. El siguiente paso es crear el botón de comando para tener la opción de editar.



image



Con esto finalizamos el proceso de edición utilizando el ObjectDataSource, el gridView y el EntityFramework.



image



Sin duda la ventaja principal con este esquema, es que podemos seguir utilizando nuestra lógica de negocios utilizando aplicaciones Web y através del ObjectDataSource no vamos a tener la necesidad de escribir código ni para seleccionar ni para actualizar datos.



Technorati Tags: ,,

11.16.2010

¿Qué paso aquí?–> Could not load type 'System.ServiceModel.Activation.HttpModule' from assembly 'System.ServiceModel, Version=3.0.0.0

Después de instalar VS 2010 y por consiguiente el framework 4.0 procedí a crear un servicio en WCF utilizando VS 2008, ya que el proyecto con el que vengo trabajando utiliza esta herramienta y el framework 3.5 para llevar a cabo sus tareas. Luego de crear el servicio, procedo a probarlo desde la consola del IIS 7 y plop!… me aparece el siguiente error en el browser.

image

Después de una búsqueda rápida me di cuenta que el problema esta en los application pools. Cómo se puede ver en la siguiente figura, la mayoría de mis sitios web corren sobre el application pool que esta por defecto en el IIS – DefaultAppPool – ya que normalmente uno no lo cambia hasta que lo va a sacar a producción { sin duda un tema completo para otro post ya que son muchas las ventajas de tener corriendo las aplicaciones en diferentes application pools}.

image

Cuando abrí la configuración del DefaultApplicationPool me di cuenta que este esta utilizando en framework 4.0 para ejecutar el proceso que administra mis sitios web.

image

Por supuesto, al estar creando yo mi aplicación en VS 2008, estoy en realidad utilizando el framework 2.0 y por lo tanto procedi a cambiarlo para que utilizara este framework.

image

y listo, despues de esto el sitio Web volvió a funcionar correctamente. Ahora, la pregunta que surge es ¿qué pasa si tengo que ejecutar aplicaciones en ambos frameworks? Pues bien, cuando se instala el Framework 4, se crean un par de application pools que tienen el framework configurado por defecto los cuales son los que se deben utilizar a la hora de utilizar el IIS. Nótese que existe uno integrado y otro clásico, esta diferencia se da por la forma en que queremos que IIS procese nuestros request, utilizando el esquema tradicional de IIS 6 [no recomendado], o el nuevo esquema de IIS 7+ [MUY RECOMENDADO]

image

A la pregunta de como cambiar el Application Pool de nuestra aplicación Web, se debe de ir a la aplicación web en el sitio web en que este creada, y en la parte derecha darle click a la opcoón Advanced Settings, seguidamente se selecciona la elipsis de Application Pool y se escoje el Application Pool deseado.

image

Technorati Tags: ,