martes, 24 de mayo de 2011

Introducción a Managed Extensibility Framework (MEF)

Qué es MEF y que te aporta

MEF es una tecnología que permite desarrollar aplicaciones extensibles. La gran ventaja que nos ofrece MEF es que no necesitamos diseñar la aplicación conociendo qué extensiones formaran parte de ella ni la implementación interna de las propias extensiones.
Por extensible nos referimos a que nuestra aplicación puede estar en producción y de forma dinámica añadir, reemplazar, eliminar las extensiones que tenemos sin necesidad ni de recompilar ni reiniciar la aplicación.
MEF viene con .NET Framework 4 en la librería: System.ComponentModel.Composition.

MEF vs IoC Container

Aunque pueda parecer que MEF e IoC  ofrecen la misma funcionalidad la diferencia consiste en que tratan de resolver problemas distintos.
El objetivo básico de un framework de IoC es el de ofrecer desacoplamiento entre componentes y nos resuelve dependencias que conocemos. Esto nos permite que nuestra aplicación sea modular y testeable.
En cambio, en MEF, el principal objetivo es la extensibilidad: el desacoplamiento es una consecuencia. Sabemos que las extensiones que se vayan a implementar cumplirán un contrato pero no sabremos si sólo habrá una extensión, muchas o ninguna.

Ejemplo: Extensión IHelloWorld

Vamos a ver un ejemplo de cómo consumir nuestras extensiones con MEF. El ejemplo es muy sencillo y únicamente tenemos una aplicación de consola que consumirá extensiones que nos devuelven el mensaje Hola Mundo en el idioma definido en la extensión.
Primero declaramos en el ensamblado Serrate.MEFDemo.HelloWorld.Interface el contrato que deben cumplir:
1namespace Serrate.MEFDemo.HelloWord.Interface
2{
3    /// <summary>
4    /// Contrato para extensiones
5    /// </summary>
6    public interface IHelloWorld
7    {
8        string GetMessage();
9    }
10}
Posteriormente en nuestra aplicación de consola Serrate.MEFDemo.HelloWorldApp tenemos la clase Consumer encargada de consumir, como no, las extensiones de terceros.
1namespace Serrate.MEFDemo.HelloWorldApp
2{
3    /// <summary>
4    /// Clase encargada de consumir nuestras extensiones
5    /// </summary>
6    public class Consumer
7    {
8        /// <summary>
9        /// Esta lista importará las extensiones que cumplan
10        /// nuestros requisitos
11        /// </summary>
12        [ImportMany]
13        public List<IHelloWorld> HelloWorldPlugins { get; set; }
14 
15        /// <summary>
16        /// Lectura de nuestras extensiones
17        /// </summary>
18        public void Read()
19        {
20            // Configuramos nuestro contenedor
21            this.Setup();
22 
23            // Recorremos la lista de extensiones
24            foreach (var plugin in HelloWorldPlugins)
25            {
26                Console.WriteLine(plugin.GetMessage());
27            }
28 
29            Console.ReadLine();
30        }
31 
32        private void Setup()
33        {
34            // a partir del directorio de la aplicación buscamos
35            // en una ruta relativa nuestro directorio de extensiones
36            string appDirectory = AppDomain.CurrentDomain.BaseDirectory;
37            DirectoryInfo extensionsDirectory =
38                new DirectoryInfo(appDirectory + "..\\..\\..\\extensions");
39 
40            // creamos el catálogo a partir del directorio
41            var catalog = new DirectoryCatalog(extensionsDirectory.FullName);
42 
43            // componemos las extensiones para poderlas usar
44            var compositionContainer = new CompositionContainer(catalog);
45            compositionContainer.ComposeParts(this);
46        }
47    }
48}
Como podemos tener varias extensiones cumpliendo el mismo contrato, en nuestra aplicación declaramos una lista de este contrato, y la decoramos con el atributo [ImportMany]:
1/// <summary>
2/// Esta lista importará las extensiones que cumplan
3/// nuestros requisitos
4/// </summary>
5[ImportMany]
6public List<IHelloWorld> HelloWorldPlugins { get; set; }
Seguidamente podemos realizar la lectura de nuestras extensiones:
1/// <summary>
2/// Lectura de nuestras extensiones
3/// </summary>
4public void Read()
5{
6    // Configuramos nuestro contenedor
7    this.Setup();
8 
9    // Recorremos la lista de extensiones
10    foreach (var plugin in HelloWorldPlugins)
11    {
12        Console.WriteLine(plugin.GetMessage());
13    }
14 
15    Console.ReadLine();
16}
En el método Setup indicamos el directorio de nuestras extensiones y realizamos la carga de éstas:
1private void Setup()
2{
3    // a partir del directorio de la aplicación buscamos
4    // en una ruta relativa nuestro directorio de extensiones
5    string appDirectory = AppDomain.CurrentDomain.BaseDirectory;
6    DirectoryInfo extensionsDirectory =
7        new DirectoryInfo(appDirectory + "..\\..\\..\\extensions");
8 
9    // creamos el catálogo a partir del directorio
10    var catalog = new DirectoryCatalog(extensionsDirectory.FullName);
11 
12    // componemos las extensiones para poderlas usar
13    var compositionContainer = new CompositionContainer(catalog);
14    compositionContainer.ComposeParts(this);
15}
Finalmente, ya sólo nos hace falta crear nuestras extensiones que realizarán el trabajo duro. Indicaremos que las extensiones son exportables mediante el atributo [Export]
Crearemos el ensamblado Serrate.MEFDemo.Plugin.ES y allí tendremos nuestra clase que implementará el contrato:
1namespace Serrate.MEFDemo.Plugin.ES
2{
3    [Export(typeof(IHelloWorld))]
4    public class HolaMundo : IHelloWorld
5    {
6        public string GetMessage()
7        {
8            return "Hola Mundo";
9        }
10    }
11}
En otro ensamblado Serrate.MEFDemo.Plugin.CA tendremos definida otra extensión:
1namespace Serrate.MEFDemo.Plugin.CA
2{
3    [Export(typeof(IHelloWorld))]
4    public class HolaMon : IHelloWorld
5    {
6        public string GetMessage()
7        {
8            return "Hola Món";
9        }
10    }
11}
Si colocamos las librerías de nuestras extensiones en la carpeta que espera nuestra aplicación, al ejecutarla ya nos saldrá por pantalla el resultado esperado:
> Hola Mundo
> Hola Món
En esta introducción hemos visto como utilizar MEF para que nuestra aplicación pueda ser extendida por nosotros mismos o por terceros de forma realmente sencilla.

No hay comentarios:

Publicar un comentario