31 de diciembre de 2014

Ejemplos de connectionStrings

Guardar la conexión en Web.config hacia SQL

   
        name="conString" connectionString="Data Source=localhost;Initial Catalog=YourDataBaseName;Integrated Security=True;" providerName="System.Data.SqlClient" />
        name="conString2" connectionString="Data Source=localhost;Initial Catalog=YourDataBaseName;Integrated Security= SSPI;" providerName="System.Data.SqlClient" />
        name="conString3" connectionString="Data Source=localhost;Initial Catalog=YourDataBaseName;Integrated Security=True;" providerName="System.Data.SqlClient" />
   

Se puede probar esta opción especificando Database en vez de Initial Catalog

 >
     name="myconnectionstring" connectionString="Server=myServerAddress;Database=myDataBase;User ID=myUsername;Password=myPassword;Trusted_Connection=False;" providerName="System.Data.SqlClient"/>
  

Y esta opción para especificar las credenciales

   name="Conn" connectionString="Data Source=192.168.1.25;Initial Catalog=Login;Persist Security Info=True;User ID=sa;Password=example.com"
   providerName="System.Data.SqlClient" />

 
miércoles, diciembre 31, 2014

WCF Data Service y Entity framework

También conocido como ADO.NET Data Services, es un componente del .NET Framework que permite crear servicios web usando ODATA (Open Data Protocol), para exponer y consumir datos en la web mediante REST, ODATA expone los datos como recursos que son accesibles y modificados a través de URIs, usando los verbos HTTP GET(Consultar), PUT(Modificar), POST(Ingresar) y DELETE(Eliminar).

La idea principal de esta entrada es exponer una colección de entidades para ser consumidas y realizar queries sobre estas por los clientes, estas entidades serán disponibles a traves de un WCF Data Service que estara hospedado en IIS, ademas se utilizara Entity Framework para construir el modelo de entidades.

Empecemos, para este ejemplo se va crear un proyecto de tipo WCF Service Application al que llamare ServicioWCF, nuestro servicio ODATA sera hospedado en el IIS, es por esto que escogemos esta opción para este ejemplo


Despues de agregar el WCF Service Application, se procede a eliminar los siguientes files pues no los utilizaremos en este ejemplo


Ahora se agregara un nuevo item  al proyecto, este item sera del tipo WCF Data Service al que llamare OData.svc


al agregar el WCF Data Service se mostrara una plantilla en la que debe especificarse en 1 la fuente de datos que el servicio utilizara para entregar información y 2 las entidades que estarán disponibles para los clientes del servicio así como los permisos que tendrán sobre esas entidades, esta información la especificaremos luego de crear el modelo de la base de datos utilizando Entity Framework


Con el paso anterior el Solution explorer del proyecto queda asi:


De nuevo agregaremos un nuevo item al proyectp, este item sera del tipo ADO.NET Entity Data Model al cual llamare  ModeloEX.edmx


El modelo se creara a partir de una base de datos existente


Guardare la conexión a la base de datos en el web.config con el nombre conexion


Las tablas agregadas se muestran en el modelo, se agregaron las tablas Employee y Project


este es el Solution Explorer del proyecto con todos los items que le agregamos en los pasos anteriores


Ahora desde nuestro modelo, damos click derecho en un espacio en blanco y seleccionamos propiedades


En propiedades buscamos el valor del Entity Container Name, en este caso es conexion


y en el código del OData.svc especificamos valores tal como se muestra en la imagen, se debe especificar el contenedor, las entidades que deseamos exponer al cliente y los derechos que el cliente tendrá sobre esas entidades


La lista con todos los permisos que se pueden otorgar a sobre una entidad se le pueden otorgar a un cliente son


Como complemento al final del código podemos agregar config.UseVerboseErrors = true; esto para lograr obtener el detalle completo de la excepción desde el web services, en caso de que ocurra algún error.

Una vez se especifican el Entity Container, los nombres de las entidades y los accesos que el cliente tendra sobre ellas, cerramos la ventana anterior y desde el Solution Explorer nos posicionamos sobre OData.svc, damos cick derecho y seleccionamos View in Browser(Default Browser), el Default Browser en mi caso es Google Chrome por eso aparece especificado en la siguiente imagen


El explorador nos mostrara la siguiente información en XML


Si queremos ver la entidad Employees en el navegador en formato AtomPub debes escribir en el navegador  http://localhost/ServicioWCF/OData.svc/Employees tambien si quieres ver esta misma informacion pero en formato JSON debes escribir en el navegador web http://localhost/ServicioWCF/OData.svc/Employees()?$format=verbosejson

Ademas, si queremos mostrar datos en especifico de la entidad Employees como por ejemplo su nombre y dirección podriamos escribir esto en el navegador

http://localhost/ServicioWCF/OData.svc/Employees?$select=nombre,direccion

El siguiente paso es hospedar este servicio en IIS (pero recuerda que puedes hospedarlos es WAS, Self Hosting o bien en un Windows Service) para eso nos posicionamos en ServicioWCF desde el Solution Explorer y damos click derecho para seleccionar propiedades

  

Escogemos la opción Web en el panel izquierdo y seleccionamos Local IIS, luego damos click sobre el botón Create Virtual Directory y deberíamos obtener un mensaje como el que se muestra en la imagen


Para verificar que en verdad se creo el directorio en IIS, abrimos el administrador de IIS escribiendo inetmgr en el buscador de Windows y deberíamos obtener la siguiente imagen con el directorio SericioWCF dentro de los Default Web Site de IIS


de nuevo vamos al Solution Explorer nos posicionamos sobre OData.svc, damos click derecho y seleccionamos View in Browser(Default Browser) y obtenemos la siguiente imagen, donde nuestra dirección web especifica que el servicio esta debidamente hospedado en localhost, que en mi caso representa al IIS instalado en mi maquina, esta dirección la seleccionamos y la copiamos pues la utilizaremos para referenciar el servicio desde nuestra aplicación cliente.


Toca crear el cliente que consumirá el WCF Data Service, como lo tengo instalado en IIS puedo referenciar desde cualquier aplicación ya sea agregada a nuestro ServicioWCF o creando una nueva aplicación, en este ejemplo creare una nueva aplicación y para ello  el Visual Studio


Creare una aplicación de Windows Forms y la llamare ProbandoData


El paso mas importante sera agregar una referencia de servicio, para eso nos posicionamos en el Solution Explorer y sobre References damos click derecho y seleccionamos Add Service Reference


ahora pegamos la direccion web (que copiamos tres pasos arriba para comprobar que nuestro servicio estaba hospedado en IIS) en la especificación de Address y presionamos Go, deberíamos obtener nuestro servicio tal como muestro en la imagen...como distintivo para hacer referencia al servicio utilizare el nombre: OdataTest


El Solution Explorer de nuestro cliente quedaría de la siguiente manera con el WCF Data Service agregado como referencia y listo para ser consumido desde nuestro Windows Form


y el diseño del formulario que voy a utilizar solo tiene un DataGridView a fin de recuperar información desde el WCF Data Service y mostrar esa información en el DataGridView


El código para el evento Load del formulario es el siguiente, primero se debe especificar la Uri que es la dirección del WCF Data Service y luego utilizar esa Uri en el constructor de ODataTest.conexion donde ODataTest es la clase y conexion el nombre del modelo de base de datos.
private void Form1_Load(object sender, EventArgs e){

  // Conectando con el WCF Data Service
  Uri acceso = new Uri("http://localhost/ServicioWCF/OData.svc/", UriKind.Absolute);
  ODataTest.conexion proxy = new ODataTest.conexion(acceso);

    var query = from emp in proxy.Employees
               select emp;
    var datos = query.ToList();
    dataGridView1.DataSource = datos;

        }
Pero solo desplegando datos no llegaremos, si desde tu formulario cliente quieren hacer uso de las entidades del modelo construido podrían utilizar este código de guía, usaremos la entidad Employees que representa a la tabla Employees así que cualquier modificación, eliminación o ingreso de datos se reflejara en esta tabla sobre la base de datos, para este ejemplo se eliminaran datos especificos segun un EmpID de la tabla Employees
try{
  
  // Conectando con el WCF Data Service
  Uri acceso = new Uri("http://localhost/ServicioWCF/OData.svc/", UriKind.Absolute);
  ODataTest.conexion proxy = new ODataTest.conexion(acceso);
  
  // Utilizando la entidad Employees desde el WCF Data Server
  ODataTest.Employees eliminarCliente = (from c in proxy.Employees        
                                         where c.EmployeesID.Equals(EmpID)
                                         select c).Single();
  
  proxy.DeleteObject(eliminarCliente);
  serv.SaveChanges();
 }

catch (Exception ex){ Throw New ArgumentExcpetion(ex.message); }

26 de diciembre de 2014

WCF - Entity Framework - AutoMapper(2)

Para realizar las operaciones CRUD se tomara como referencia el proyecto creado en la entrada anterior, pueden consultarlo en este enlace
La interfaz ILunesService.cs se modificara de manera que muestre mas metodos al cliente, queda asi:


Tambien se modifica la clase LunesService.svc.cs que implementa la interfaz anterior de la siguiente manera


es todo del lado del servicio WCF, ahora del lado del cliente se modificara el formulario que se muestra al usuario


y el codigo para los eventos click de los botones se muestra a continuacion


 



WCF - Entity Framework - AutoMapper(1)

Automapper sirve para hacer mapeos entre objetos .NET, ahorrandonos gran cantidad de código y que provee algunas utilidades para personalizar cómo se realiza dicho mapeo entre objetos. Realizar mapeo entre objetos en una funcionalidad común en muchos de nuestros desarrollos, ya que permiten personalizar la salida de los datos evitando revelar información confidencial y dando al cliente solo la información que él realmente necesita.

En esta entrada se muestra el uso de la funcion mas basica de AutoMapper: Flattening, para eso instalamos el automapper desde la consola de Nuget escribiendo: Install-Package AutoMapper -Version 3.3.0 una vez instalado creamos un servicio de WCF, le agregamos un modelo de base de datos utilizando el ORM Entity Framework y separamos las entidades creadas por el EF en una biblioteca de clases, tal como se desarrollo en este enlace

El modelo de la base de datos tiene las siguientes entidades

Pero no queremos mandar todos las propiedades de esta entidad, ya que el campo CreadoPor y FechaCreacion sirven para control interno ademas que los datos ingresados deben ser validados para mantener la integridad de la base de datos, por lo que la entidad que nos interesa mandar al cliente solo deberia contener las propiedades CodigoEmp, NombreEmp y CargoEmp, es para esto que se utilizara el AutoMapper, para crear un objeto con las propiedades que me interesan(al que llamare EmployeeDTO) a partir de otro objeto similar(Employee)

Para esto se agregaran dos clases a la bilbioteca de clases: EmployeeDTO y ProjectDTO, el solution Explorer tendra dos proyectos: lunes(que tendra el modelo de base de datos y el servicio WCF a hospedar en IIS) y la bilbioteca de clases: Clases(que tendra las entidades creadas por el EF en Modelo.tt y las clases DTO que nosotros crearemos), queda de la siguiente manera:


la clase EmployeeDTO  de nuestra autoria y la clase Employee generada por el ORM ENtity Framework quedan asi:

EmployeeDTO.cs

Employee.cs

Con esto ya tenemos dos objetos que no comparten todas las propiedades pero si algunas, es importante aclarar que la configuración de las transformaciones solo hay que realizarla una vez por AppDomain por eso ahora debemos agregar dos clases mas al proyecto: AutomapBootstarp.cs y AutomapServiceBehaviour.cs que se muestran a continuacion, fijarse en las referencias y espacios de nombre utilizados

AutomapBootstarp.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using Clases;
using AutoMapper;

namespace lunes{
    public class AutomapBootstrap{
        public static void InitializeMap(){
            Mapper.CreateMap<Employee, EmployeeDTO>().ReverseMap();
           
        }}}


AutomapServiceBehaviour.cs
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Linq;
using System.ServiceModel;
using System.ServiceModel.Channels;
using System.ServiceModel.Description;
using System.Web;

namespace lunes{
    public sealed class AutomapServiceBehavior : Attribute, IServiceBehavior{
        public AutomapServiceBehavior(){}

        public void AddBindingParameters(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase,
            Collection<ServiceEndpoint> endpoints, BindingParameterCollection bindingParameters)
        {
            AutomapBootstrap.InitializeMap();
        }

        public void ApplyDispatchBehavior(ServiceDescription serviceDescription, System.ServiceModel.ServiceHostBase serviceHostBase)
        { }

        public void Validate(ServiceDescription serviceDescription, System.ServiceModel.ServiceHostBase serviceHostBase)
        { }

    }}
 

Ahora el desarrollo de la interfaz del servicio, lo unico que hara es enviar un listado de EmployeeDTO para que sea mostrado en el DataGridView de un cliente, la interfaz ILunesService.cs y de la clase que lo implementa LunesService.svc.cs se muestra

ILunesService.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.Text;
using Clases;
using System.Collections;

namespace lunes{
 
    [ServiceContract]
    public interface ILunesService{
        [OperationContract]
        List<EmployeeDTO> Listado();       
    }}
 

En la clase LunesService.svc.cs se debe colocar la etiqueta [AutomapServiceBehavior]
LunesServicio.svc.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.Text;
using AutoMapper;
using System.Data.Entity;
using Clases;
using System.Web;

namespace lunes{
      [AutomapServiceBehavior]
   // [ServiceErrorBehavior(typeof(ElmahErrorHandler))]
   // [AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]
    public class LunesService : ILunesService
    {
    public LunesService(){}
       
        public List<EmployeeDTO> Listado(){
            conexion cnn = new conexion();
            var datos = (from emp in cnn.Employees
                         select emp).ToList();
            List<EmployeeDTO> lista = Mapper.Map<List<Employee>, List<EmployeeDTO>>(datos);
            return lista;
       
    }}}

Ahora se hospedara el servicio en IIS tal como se muestra en este enlace

El siguiente paso es crear un cliente que consumira el servicio, para este ejemplo se debe abrir Visual Studio y crear una aplicacion Windows Forms con un formulario y un DataGridView, se hace la referencia al servicio al cual llamare a su referencia zerbixio(para referencia este enlace) y este sera el codigo para cargar el DataGridView con los datos que tenga EmployeeDTO

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

namespace datosEJ{
    public partial class Form1 : Form{
        public Form1(){  InitializeComponent();  }

        private void Form1_Load(object sender, EventArgs e){
            try{
                zerbixio.LunesServiceClient clase = new zerbixio.LunesServiceClient();
                dataGridView1.DataSource = clase.Listado();
            }
            catch (Exception ex){ MessageBox.Show(ex.Message); }
          
        }}}

con esto se obtiene el resultado deseado


Datos complementarios: uso del ForMember cuando los objetos tienen el mismo tipo de datos y hacen alusion al mismo dato pero se denominan diferente(primeras cuatro lineas del ejemplo) o bien se quiere la combinacion de valores desde el objeto fuente(ultima linea del ejemplo, en este ultimo caso se puede agregar una propiedad nueva al objeto destino, sin necesidad que haga referencia a alguna propiedad en el objeto origen, por ejemplo Double CostaAlCliente no existe en la clase origen sino mas bien se agrego de tipo Double para mostrar una combinacion de valores de la clase origen.

Mapper.CreateMap<Empleado, EmpleadoOferta>()
                .ForMember(dest => dest.Nombre, fte => fte.MapFrom(src => src.Nombre))
                .ForMember(dest => dest.Id, fte => fte.MapFrom(src => src.Id))
                .ForMember(dest => dest.AnnosExperiencia, fte => fte.MapFrom(src => src.Experiencia))
                .ForMember(dest => dest.Especialidad, fte => fte.MapFrom(src => src.Especialidad))
                .ForMember(dest => dest.CostoAlCliente, fte => fte.MapFrom(src => (src.SalarioXHora * 0.40) + src.SalarioXHora));