Renaud Harduin
Optimisation des cubes par design d'agrégations
J'ai toujours eu un peu de mal à m'en remettre totalement aux algorithmes interne de SSAS et je vous propose au travers de cet article de choisir les patterns d'agrégation et de les pousser avec AMO
Par Renaud Harduin publié le 09/03/2008 à 21:59
 
AMO
AMO signifie Analysis Services Object.Je vous renvoie déjà sur la MSDN : http://msdn2.microsoft.com/en-us/library/ms124924.aspx
Globalement, nous disposons d'une classe Server sur la quelle nous établissons une connexion.
Sur ce Server, nous retrouvons une Collection de Database.La Database contient une collection de cube qui parente une collection MeasureGroup.Chaque MeasureGroup référence nos AgregationDesign.
Pour comprendre le fonctionnement, je vous propose de construire un outil simple qui va nous ressortir sous forme tabulaire (XML -> Excel) une matrice des attributs et de leur contribution aux agrégats (à l'image du designer sous SS2K8 que nous n'avons pas sous SS2K5 ).
Je vous propose donc de créer une solution qui va nous permette de générer une cartographie de nos agrégations.
 
/content/76863bb1-4314-4ec4-bcb4-644295bac3d3/image006.jpg
 
Il est tout d'abord nécessaire de faire référence à AMO. L'assembly se trouve sous : C:\Program Files\Microsoft SQL Server\100\SDK\Assemblies\Microsoft.AnalysisServices.dll (la référence est la même sur SSAS 2005) et référence le namespace : Microsoft.AnalysisServices
Je définis de manière globale au projet :

private string _connString = @"Provider=MSOLAP.4;Integrated Security=SSPI;Persist Security Info=False;

Initial Catalog=Adventure Works DW Standard Edition;Data Source=LOCALHOST";

Server server;

Je reproduis ici de manière un peu mécanique le code de mon IHM :

public Form1()

{

    server = new Server();

    server.Connect(_connString);

    InitializeComponent();  

        }

 

private void button1_Click(object sender, EventArgs e)

{

    foreach (Database db in server.Databases)

    {

        cbxDatabase.Items.Add(db.Name);             

    }       

}

 

private void btnCube_Click(object sender, EventArgs e)

{

    CubeCollection cubes = server.Databases[cbxDatabase.Text].Cubes;

    foreach (Cube cube in cubes)

    {

        cbxCube.Items.Add(cube.ID);

    }

}

 

private void btnPartition_Click(object sender, EventArgs e)

{

    Cube cube = server.Databases[cbxDatabase.Text].Cubes[cbxCube.Text];

    AggregationDesignCollection adcoll = 

        cube.MeasureGroups[cbxMeasuresGroup.Text].AggregationDesigns;

 

    foreach (AggregationDesign ad in adcoll)

    {

        cbxAggDesign.Items.Add(ad.ID);

    }          

}

 

private void btnMeasureGroup_Click(object sender, EventArgs e)

{

    MeasureGroupCollection mc =

          server.Databases[cbxDatabase.Text].Cubes[cbxCube.Text].MeasureGroups;

 

    foreach ( MeasureGroup mg in mc)

    {

        cbxMeasuresGroup.Items.Add(mg.ID);

 

    }

}

Lorsque le projet se charge, nous établissons une connexion au serveur OLAP (server.connect() )L'utilisateur alimente les combos en cliquant les boutons :
  • Select db (button1_Click) permet d'alimenter la liste des bases disponibles pour la connection
  • Le bouton “Select Cube” permet d'alimenter la liste des cubes pour la Database sélectionnée (btnCube_Click).
  • Une fois le cube sélectionné, il faut identifier les groupes de mesures dispobible (btnMeasureGroup_Click)
  • Enfin, une fois le “MeasureGroup” choisis, le dernier bouton permet d'alimenter la liste des AggregationDesign (btnPartition_Click) .
Lors du clic sur le bouton de sauvegarde du plan de stockage, nous récupérons les références au cube d'une part et aux AggregationsDesign sélectionné d'autre part:

private void btnSave_Click(object sender, EventArgs e)

{

    StringBuilder sb = new StringBuilder();

 

    Cube cube = server.Databases[cbxDatabase.Text].Cubes[cbxCube.Text];

    AggregationDesign ad =

        cube.MeasureGroups[cbxMeasuresGroup.Text].AggregationDesigns[cbxAggDesign.Text];

Mon objectif est de générer un enregistrement par Agrégation, en sauvegardant dans un fichier xml dont on attend par exemple :

<?xml version="1.0" encoding="utf-8"?>

<Aggregations>

  <Aggregation>

    <OrderNumber>0</OrderNumber>

    <Name>Aggregation_0</Name>

    <Fact_Internet_Sales></Fact_Internet_Sales>

    <Dim_Customer></Dim_Customer>

    <Product>Model_Name Days_To_Manufacture </Product>

    <Promotion>Promotion_Category </Promotion>

    <Dim_Sales_Territory></Dim_Sales_Territory>

    <Dim_Currency></Dim_Currency>

  </Aggregation>

 

  …

 

  <Aggregations>

A cet effet on va donc itérer dans les Agrégations :

for (int i = 0; i < ad.Aggregations.Count; i++)

    {

        Aggregation ag = ad.Aggregations[i];

 

 

        sb.AppendLine("<Aggregation>");

 

        sb.AppendLine("<OrderNumber>");

        sb.AppendLine(""+i);

        sb.AppendLine("</OrderNumber>");

 

 

        // Aggregation Name

        sb.AppendLine("<Name>");

        sb.AppendLine(ag.Name.Replace(" ", "_"));

        sb.AppendLine("</Name>");

A ce niveau, il est nécessaire d'aller explorer pour chaque Agrégation ses dimensions. Enfin, pour une dimension donnée, on pourra connaitre quels sont les attributs impliqués dans le stockage en explorant la collection Attributes conformément aux API

foreach (AggregationDimension agDim in ag.Dimensions)

{

    sb.AppendLine("<" + agDim.CubeDimensionID.Replace(" ", "_") + ">");

    foreach (AggregationAttribute attrib in agDim.Attributes)

    {

        sb.Append(attrib.AttributeID.Replace(" ", "_"));

        sb.Append(" ");

    }

 

    sb.AppendLine("</" + agDim.CubeDimensionID.Replace(" ", "_") + ">");

}

sb.AppendLine("</Aggregation>");

Il ne reste plus alors qu'à sauvegarder le XML et à le récupérer dans Excel :

    sb.AppendLine("</Aggregations>");

 

    System.IO.StreamWriter file = new System.IO.StreamWriter( tbxFile.Text);

    file.WriteLine(sb.ToString());

    file.Close();

 

    MessageBox.Show("File Saved");          

 

}

Une fois ouvert dans Excel :
 
/content/76863bb1-4314-4ec4-bcb4-644295bac3d3/image007.jpg
 
Pour chaque ligne, nous trouvons un agrégat et sa description. Pour chaque dimension, vous trouverez les attributs sur lesquels SSAS a choisis de consolider.

 Commentaire - Optimisation des cubes par design d'agrégations 

 Dernières Publications      

Utilisation de jQuery avec ASP.NET MVC
  Développer une IHM à page unique avec ASP.NET MVC et jQuery
par Nicolas Moyère posté le 30/06/2008 à 10:28, lu 824 fois, #0
Tags: ASP.NET MVC, Ajax
Windows Media Center et WCF : développez votre maison intelligente
  Le développement d'applications pour Windows Media Center est facilité avec l'arrivée du SDK 5.3. Même si l'on sent un modèle objet bien lourd derrière, il devient plus facile d'exposer les fonctionnalités de WMC sous la forme de services WCF.
par Frédéric Colin posté le 23/06/2008 à 08:04, lu 891 fois, #0
Notions avancées avec Biztalk Server 2006 R2
  Utilisation des notions d'interchange, corrélation et convoi avec BizTalk Server 2006 R2
par Kader Yildirim posté le 09/06/2008 à 08:04, lu 705 fois, #0
Lucene Persistence Engine pour Evaluant Universal Storage Services
  Suite à l'article de Laurent Kempé, voici un moteur de stockage pour EUSS permettant l'indexation d'entités métier avec Lucene.
par Nicolas Penin posté le 01/06/2008 à 23:38, lu 1091 fois, #1
Tags: C#, Linq
XMLA Trivia : Découverte du XMLA
  Le XMLA (XML for Analysis) est un langage normalisé par plusieurs éditeurs BI pour simplifier l'accès aux données aux cubes et aux métadonnées des bases multidimensionnelles.
par Renaud Harduin posté le 25/05/2008 à 11:57, lu 1008 fois, #1
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 2784 fois, #0
Comment manipuler simplement le contenu d'un fichier WordML ?
  Manipulations autour du format WordML
par Fabien Reinle posté le 14/05/2008 à 23:55, lu 1405 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 2931 fois, #2

 Dernières Actualités      

Reprise du projet Reflector par RedGate
  La nouvelle était connue depuis quelques jours par les développeurs de plugins, mais c’est désormais officiel : Lutz Roeder, le responsable de Reflector confie à la société RedGate le futur du projet....
Microsoft publie Visual Studio 2008 Service Pack 1
  Il est recommandé d’utiliser l’outil Visual Studio 2008 Service Pack preparation Tool avant de faire l’installation du Service Pack si vous avez installé des versions béta sur votre machine. Une fois que...
Tags: Framework .NET, Visual Studio 2008
Evaluant dévoile ses sources
  L'ensemble des projets R&D réalisés par les consultants de la SSII Evaluant sont en cours de publication sur CodePlex . L'objectif est de les centraliser et surtout d'augmenter leur visibilité. L'avantage...
Le Silverlight Tour en français!
  Le Silverlight Tour passe maintenant dans les pays francophones! En effet RunAtServer Consulting est partenaire du Silverlight Tour pour la gestion de cette formation Silverlight en français à commencer...
Microsoft publie ASP.NET AJAX 4.0 CodePlex Preview 1
  Cette pré-version contient les améliorations suivantes: Client-side template rendering Declarative instantiation of behaviors and controls DataView control Markup extensions Bindings Vous pouvez en lire...
Tags: Ajax
Deep Earth – Une belle utilisation de Virtual Earth et de Silverlight Deep Zoom
  Ce projet très intéressant est disponible sur Codeplex et vous pouvez voir une démo sur la page suivante . Bien entendu comme touts les projets sur Codeplex vous avez accès aux sources....
Tags: Silverlight