tag(s) Tags: C#, ASP.NET 2
lu 789 fois
6 pages
Jean-Marie Thia
Module en code managé pour IIS7 - 1
Utilisation de l'interface d'administration d'IIS7 pour manipuler des modules http écrit en code managé et portage d'un module en code managé pour intégration dans le système de gestion de configuration d'IIS 7
Par Jean-Marie Thia publié le 08/02/2008 à 09:50
 
Tous les paramètres sont maintenant en place, il ne reste plus qu'à modifier le code du module pour les récupérer. Pour cela nous allons appeler à la rescousse les API de Microsoft.Web.Administration.
Nous allons commencer par écrire une classe CasAuthenticationConfigurationSection qui nous permettra de typer et manipuler les attributs de la section.

public class CasAuthenticationConfigurationSection : ConfigurationSection

{

    public static readonly string SectionName = "system.webServer/casAuthentication";

 

    public bool Enabled

    {

        get { return (bool)base["enabled"]; }

        set { base["enabled"] = value; }

    }

 

    public string LoginUrl

    {

        get { return (string)base["loginUrl"]; }

        set { base["loginUrl"] = value; }

    }

 

    public string ValidateUrl

    {

        get { return (string)base["validateUrl"]; }

        set { base["validateUrl"] = value; }

    }

 

    public string LogoutUrl

    {

        get { return (string)base["logoutUrl"]; }

        set { base["logoutUrl"] = value; }

    }

Il faut, peut être, ajouter Microsoft.Web.Application dans les références de l'application. Pour information la dll se trouve dans %windir%\system32\inetsrv sur mon serveur Windows 2008 build 6001.
Pas de grosse modification pour le code, il faut juste instancier un objet CasAuthenticationConfigurationSection de la manière suivante :

CasAuthenticationConfigurationSection configuration = (CasAuthenticationConfigurationSection)

WebConfigurationManager.GetSection(

    HttpContext.Current,

    CasAuthenticationConfigurationSection.SectionName,

    typeof(CasAuthenticationConfigurationSection));

// skip all if the module is not enabled

if (!configuration.Enabled)

    return;

Attention à ne pas oublier de ne rien faire si le module n'est pas activé comme le montre les deux dernières lignes du code précédent.
La lecture des attributs est triviale grâce au wrapper

// retrieve the cas host urls from web configuration file

string casLogin = configuration.LoginUrl;

string casValidate = configuration.ValidateUrl;

et c'est fini, enfin presque...
Il faut signer la dll avant sa compilation car elle doit être installée dans le GAC.

private void Application_AuthenticateRequest(Object source, EventArgs e)

{

    CasAuthenticationConfigurationSection configuration = (CasAuthenticationConfigurationSection)

    WebConfigurationManager.GetSection(

        HttpContext.Current,

        CasAuthenticationConfigurationSection.SectionName,

        typeof(CasAuthenticationConfigurationSection));

    // skip all if the module is not enabled

    if (!configuration.Enabled)

        return;

 

    // get a HttpApplication objects to gain access request and response properties.

    HttpApplication application = (HttpApplication)source;

 

    // retrieve the cas host urls from web configuration file

    if (String.IsNullOrEmpty(configuration.LoginUrl))

    {

        // trigger a server error if cashost is not set in the web.config

        throw new HttpException(500, "CasAuthentication module misconfigured : missing LoginUrl");

    }

    string casLogin = configuration.LoginUrl;

 

    if (String.IsNullOrEmpty(configuration.ValidateUrl))

    {

        // trigger a server error if cashost is not set in the web.config

        throw new HttpException(500, "CasAuthentication module misconfigured : missing ValidateUrl");

    }

    string casValidate = configuration.ValidateUrl;

 

 

 

    // Create a generic principal on each request based on the authentication ticket cookie

    // The user's role is also extracted from this cookie and pushed into the generic principal

    string cookieName = FormsAuthentication.FormsCookieName;

    HttpCookie authCookie = application.Request.Cookies[cookieName];

 

    // if the cookie exist we are authenticated

    if (authCookie != null)

    {

        FormsAuthenticationTicket authTicket = null;

        try

        {

            authTicket = FormsAuthentication.Decrypt(authCookie.Value);

        }

        catch

        {

            // TODO: Make a 500 error or go back to authentication

            return;

        }

 

        if (authTicket == null)

        {

            throw new HttpException(401, "");

        }

 

        // create an identity objet

        FormsIdentity identity = new FormsIdentity(authTicket);

        // create a principal

        GenericPrincipal principal = new GenericPrincipal(identity, null);

        // attach the principal to tue context objet that will flow throughout the request.

        application.Context.User = principal;

    }

    else

    {

        // Check if we are back from CAS Authentication

 

        // Look for the "ticket=" string after the "?" in the URL when back from CAS

        string casTicket = application.Request.QueryString["ticket"];

 

        // The CAS service name is the page URL for CAS Server call back

        // so any query string is discard.

        string service = application.Request.Url.GetLeftPart(UriPartial.Path);

 

        // First pass because there is no ticket, so redirect to CAS login

        if (casTicket == null || casTicket.Length == 0)

        {

            // memorize the initial request query string

            application.Response.Cookies[ReturnUrl].Value = application.Request.RawUrl;

            // redirect to cas server

            string redir = casLogin + "?service=" + service;

            application.Response.Redirect(redir);

            return;

        }

        else

        {

            // Second pass (return from CAS server) because there is a ticket in the query string to validate

            string validateurl = casValidate + "?ticket=" + casTicket + "&" + "service=" + service;

            WebClient client = new WebClient();

            StreamReader Reader = new StreamReader(client.OpenRead(validateurl));

 

            // Put the validation response in a string

            string resp = Reader.ReadToEnd();

 

            // Some boilerplate to set up the parse of validation response.

            NameTable nt = new NameTable();

            XmlNamespaceManager nsmgr = new XmlNamespaceManager(nt);

            XmlParserContext context = new XmlParserContext(null, nsmgr, null, XmlSpace.None);

            XmlTextReader reader = new XmlTextReader(resp, XmlNodeType.Element, context);

 

            string netid = null;

 

            // A very dumb use of XML by looping in all tags.

            // Just scan for the "user". If it isn't there, its an error.

            while (reader.Read())

            {

                if (reader.IsStartElement())

                {

                    string tag = reader.LocalName;

                    if (tag == "user")

                    {

                        netid = reader.ReadString();

                        break;

                    }

                }

            }

            reader.Close();

 

            // If there was a problem, leave the message on the screen. Otherwise, return to original page.

            if (netid == null)

            {

                application.Response.Write("Votre identité n'a pas été validé par le serveur CAS");

                application.Response.Write(resp);

            }

            else

            {

                application.Response.Write("Bienvenue " + netid);

 

                // create the authentication ticket and store the roles in the user data

                FormsAuthenticationTicket formAuthTicket = new

                    FormsAuthenticationTicket(

                            1,                              // version

                            netid,                          // user name

                            DateTime.Now,                   // creation

                            DateTime.Now.AddMinutes(20),    // expiration

                            false,                          // persistant

                            "");                            // userData

 

                // encrypt the ticket

                string encryptedTicket = FormsAuthentication.Encrypt(formAuthTicket);

                // create a cookie and use the encrypted ticket as data

                authCookie = new HttpCookie(FormsAuthentication.FormsCookieName, encryptedTicket);

                // add the cookie to the response cookie collection

                application.Response.Cookies.Add(authCookie);

 

                // go the initial request URL

                string returnUrl;

                // if the return url cookie is lost, return to the default page

                if (application.Request.Cookies[ReturnUrl] == null)

                    returnUrl = application.Request.ApplicationPath;

                else

                    returnUrl = application.Request.Cookies[ReturnUrl].Value;

 

                application.Response.Redirect(returnUrl);

            }

        }

    }

}

 Commentaire - Module en code managé pour IIS7 - 1 

 Dernières Publications      

Exploiter les données CSV via Linq en toute simplicité
  A partir du requêteur dynamique fourni en exemple avec Visual Studio 2008, nous allons essayer de remplir les propriétés d'un ensemble d'objets à partir des données d'un fichier CSV. Nous enrichirons aussi le parseur de nos propres fonctions.
par Frédéric Mélantois posté le 17/05/2008 à 11:41, lu 140 fois, #0
Polymorphisme et contrats de données WCF
  WCF aborde les types polymorphes du point de vue de la sérialisation. En effet, la connaissance du type réel potentiel est rendue nécessaire dès la description du contrat de données. Une fois n'est pas coutume, j'ai réalisé l'exemple en VB.NET.
par Frédéric Colin posté le 14/05/2008 à 08:40, lu 296 fois, #2
A la découverte de BizTalk Server 2006 3/3
  Développer un assembleur pour BizTalk Server 2006 R2
par Kader Yildirim posté le 06/05/2008 à 13:20, lu 147 fois, #0
Chapitre III : Sync Services for ADO.NET et WCF
   Suite des deux premiers chapitres sur la synchronisation avec Sync Services for ADO.NET, voici un nouvel article impliquant WCF dans une synchronisation déconnectée.
Requêtes dynamiques sur les IEnumerable
  A partir d'un exemple fourni avec Visual Studio 2008, initialement prévu pour tout objet Queryable, nous allons présenter comment en ajoutant très peu de code rendre disponible aux IEnumerable un requêteur dynamique.
par Frédéric Mélantois posté le 24/04/2008 à 15:03, lu 842 fois, #0
Développer une visionneuse d'images avec WPF et WCF
  Au travers de cet article, nous allons découvrir comment mettre en place une visionneuse d'images, grâce aux technologies WPF et WCF.
par Thomas Lebrun posté le 22/04/2008 à 22:46, lu 1005 fois, #2
LINQRoleProvider
  L'objectif de cet article est d'implémenter un fournisseur de rôles ASP.NET personnalisé à l'aide de LINQ To Sql tout en faisant un tour d'horizons de la syntaxe des requêtes LINQ.
par Antoine Griffard posté le 13/04/2008 à 22:18, lu 491 fois, #4
WCF : Transfert de messages streamés et sécurisation personnalisée
  Je poursuis ma série d'articles sur WCF en vous présentant cette fois-ci le mode de communication Streamé. Histoire d'aller un petit peu plus loin, j'ai protégé le service de manière personnalisée et utilisé un binding très courant : BasicHttpBinding
par Frédéric Colin posté le 07/04/2008 à 08:12, lu 884 fois, #0

 Dernières Actualités      

Injection de code et API de profiling .NET
  Si vous êtes intéressés par la sécurité du Framework, par le reverse engineering et la manipulation/injection de code .NET et les packers, alors jetez un coup d’œil...
NDepend pour l'analyse statique de code .NET
  Pour ceux qui ne connaissent pas NDepend , il s’agit d’un outil d’analyse statique de code .NET qui permet de remonter des informations à toute une équipe de développement. NDepend aide à travailler sur...
Tags: Outils
Microsoft met à disposition son IoC Container Unity 1.0 en version finale
  Microsoft met à disposition la version 1.0 de son IoC container Unity, sur CodePlex sous la forme d'un Application Block des Enterprise Library. Si vous voulez en savoir plus sur le sujet en .NET je vous...
Tags: Application Block
Microsoft MVP (Most Valuable Professional) sur Tech Head Brothers
  Je voudrais féliciter les nouveaux Microsoft MVP (Most Valuable Professional) du mois d'Avril 2008 qui publient sur Tech Head Brothers! Sans les auteurs le site ne serait rien. Sébastien Pertus - MVP SQL...
Les Webcasts des Microsoft TechDays 2008 sont en ligne
  Si vous avez participé ou non au Microsoft TechDays 2008 vous avez certainement manqué certaines présentation qui vous intéressaient. Voilà enfin votre chance de pouvoir suivre ces présentations en ligne...
Rapide résumé de la conférence MIX08
  La conférence MIX08 s'est achevée la semaine dernière avec peu d'annonces de nouvelles technologies (DeepZoom) mais surtout des releases de produits (et c'est pas un mal...). Nous avons donc eu droit aux...