1.4 Détails des sources de données RDF

Écrit par Neil Deakin. Traduit par René-Luc D'Hont (13/06/2005).
Page originale : http://www.xulplanet.com/tutorials/mozsdk/rdfsrcdetails.php xulplanet.com

Cette section entre dans le détail de certains des datasources de base fournis par Mozilla.

Datasources de base

Comme décrit dans la section précédente, Mozilla fournit trois formes de datasource de base. Ces datasources de base sont le in-memory-datasource, le xml-datasource et le composite-datasource. Ceux-ci sont décrits plus en détail ci-dessous. En outre, divers datasources sont fournis qui sont utilisés pour stocker des types particuliers de données. Par exemple, Mozilla fournit des datasources pour les signets (bookmarks), l'historique, les moteurs de recherche installés, les dossiers et ainsi de suite. Certains de ces datasources ne sont que des empaquetages des trois types de base. Par exemple, le datasource des signets est un emballage autour d'un in-memory-datasource. Quand on modifie les signets, il finit réellement par appeler le in-memory-datasource interne pour enregistrer les données. Ceci est fait parce que ce datasource intérieur n'a aucune limitation sur le genre de données qu'il peut stocker. Cependant le datasource de signets a besoin d'assurer que les données saisies sont valides dans le contexte des signets. D'autres datasources, tels que le datasource d'historique, ont des emballages autour d'autres structures de données internes.

Memory datasources

Le in-memory-datasource contient toutes les données RDF en mémoire. Il a été conçue pour être efficace en interrogation et en modification de données. C'est le type de datasource que vous emploierez pour les données faites sur commandes qui ne sont pas chargées à partir d'un fichier RDF/XML.

Le memory datasource applique toutes les méthodes de l'interface nsIRDFDataSource comme pour tous les datasources. Il implémente aussi l'interface nsIRDFInMemoryDataSource qui fournit une méthode simple EnsureFastContainement (assurer un contenu rapide). Vous n'appellerez normalement pas cette méthode vous-même. Elle est utilisé intérieurement pour optimiser le stockage. Fondamentalement, elle est utilisé pour assurer qu'une ressource particulière dans le datasource soit stockée de façon à y avoir accès plus rapidement. La différence est que plus de mémoire est employée. Les conteneurs RDF le font automatiquement quand ils ont un grand nombre d'enfant, car sinon l'accès serait plus lent.

XML datasources

Le xml-datasource est employé pour des données chargées à partir d'un fichier RDF/XML. Ces fichiers peuvent être des fichiers locaux ou des fichiers distants stockés sur un site Web. Ces datasources peuvent être aussi bien rechargés et sauvés. Charger un fichier RDF/XML est simple comme appeler la méthode GetDataSource du service RDF. Ceci chargera le fichier et l'parsera dans un datasource. Le xml-datasource n'est réellement qu'un paquetage autour du in-memory-datasource. Ce datasource intérieur contient les données du RDF.

Le fichier RDF/XML doit avoir un type de contenu HTTP text/XML, application/XML ou text/rdf. Mozilla ne supporte pas actuellement les RDF/XML déclaré comme application/rdf+XML. Un RDF/XML déclaré d'une autre manière que celle ci-dessus ne sera pas chargé dans un datasource. Quand vous chargez des datsources en utilisant le service RDF, vous devez toujours spécifier l'URL absolue du fichier RDF/XML, pas une URL relative.

Les datasources RDF/XML peuvent être chargés de n'importe quel type d'URL. Actuellement, seulement ceux chargés à partir d'une URL de fichier (les URLs qui commencent par « file: ») peuvent être modifier avec les APIs RDF de modification. Une façon de contourner le problème de modification des sources RDF distantes est de charger le RDF et ensuite d'ajouter les données dans un in-memory-datasource séparé.

La méthode GetDataSource charge le fichier de façon asynchrone. Cela signifie que le datasource peut ne pas être chargé quand la méthode répond. Un chargement synchronisé peut être effectué à l'aide de la méthode GetDataSourceBlocking. Cette méthode attendra jusqu'à la fin du chargement avant de renvoyé quelque chose. Notez que ceci semblera retenir l'interface utilisateur le temps que le datasource soit chargé.

Il peut être utile d'employer le chargement asynchrone pour pouvoir déterminer quand le datasource a été chargé. Le xml-datasource implémente une interface nsIRDFXMLSink qui est une interface d'aide appelée pendant le chargement et l'analyse du RDF. La plupart du temps il est employé intérieurement, mais l'interface peut être employé pour ajouter un observateur qui annoncera quand le RDF/XML aura été chargé. Voici un exemple:


var observer = {
  onBeginLoad : function(sink){},
  onInterrupt : function(sink){},
  onResume : function(sink){},
  onError : function(sink,status,msg){},

  onEndLoad : function(sink){
    sink.removeXMLSinkObserver(this);
    sink.QueryInterface(Components.interfaces.nsIRDFDataSource);
  }
};

var ds=rdfService.GetDataSource("http://www.xulplanet.com/tutorials/xultu/animals.rdf");
ds.QueryInterface(Components.interfaces.nsIRDFXMLSink);
ds.addXMLSinkObserver(observer);

la méthode addXMLSinkObserver est employé pour ajouter un observateur du processus de chargement du datasource. L'observateur doit utilisé l'interface nsIRDFXMLSinkObserver qui dans l'exemple est utilisé par un objet Javascript. La méthode onEndLoad est d'un intérêt particulier, elle sera appelée quand les données auront été entièrement chargée. Dans cette méthode, nous saisissons l'occasion d'enlever l'observateur en utilisant la méthode removeXMLSinkObserver. C'est là que vous mettriez le code à exécuter après que le datasource ait été chargé. Note that the datasource may be casted to and from the nsIRDFXMLSink directly.

le xml-datasource implémente aussi l'interface nsIRDFRemoteDataSource. En dépit du nom, cette interface contient des méthodes pour recharger et sauver le datasource. La sauvegarde d'un datasource sera discutée dans une prochaine section.

Pour recharger un datasource, utilisez la méthode Refresh. Elle prend un argument, si elle doit bloquer alors que ça charge ou pas. Si c'est true, la méthode attendra que le datasource ait été entièrement chargé avant le renvoi. Si c'est false, la méthode répondra immédiatement. L'exemple ci-dessous obtiendra un datasource et le rechargera.


var ds=rdfService.GetDataSource("file:///main/data/animals.rdf");
ds.QueryInterface(Components.interfaces.nsIRDFRemoteDataSource);
ds.Refresh(true);

Analyser (parser) un RDF/XML comme une chaîne de caractère

Vous pouvez également analyser un RDF/XML comme une chaîne de caractère. Ceci implique de créer un composant d'analyse de RDF/XML et d'assurer un datasource pour l'analyser. Le xml-datasource a la capacité d'analyser à l'intérieur de lui-même, mais le parseur peut ajouter des données à n'importe quel datasource modifiable. Le parseur emploie les méthodes de modification de l'interface nsIRDFDataSource, donc tout ce qui est exigé c'est un datasource qui manipule ces méthodes convenablement.

L'interface nsIRDFXMLParser a une méthode qui peut être employée pour analyser une chaîne de caractère RDF/XML. Dans cet exemple, nous analysons les données dans un in-memory-datasource.


function parseRDFString(str, url)
{
  var memoryDS = Components.classes["@mozilla.org/rdf/datasource;1?name=in-memory-datasource"]
                   .createInstance(Components.interfaces.nsIRDFDataSource);

  var ios=Components.classes["@mozilla.org/network/io-service;1"]
                  .getService(Components.interfaces.nsIIOService);
  baseUri=ios.newURI(url,null,null);

  var parser=Components.classes["@mozilla.org/rdf/xml-parser;1"]
                       .createInstance(Components.interfaces.nsIRDFXMLParser);
  parser.parseString(memoryDS,baseUri,str);

  return memoryDS;
}

Cette fonction peut être découpée en trois parties. La première partie crée un nouveau datasource vide. La deuxième partie crée un objet Uri, puisque tous les datasources doivent avoir un URI. Puisque nous analysons une chaîne de caractère, nous n'avons aucun URI spécifique, ainsi nous pourrions juste en composer un. Toutes les références relatives dans le RDF/XML seront résolues relativement à cet URI. En conclusion, le parseur de RDF/XML est créé et nous analysons le contenu en utilisant la méthode parseString. Cette méthode prend trois arguments, le datasource à analyser, la base URI créée dans la deuxième partie et la chaîne de caractère à analyser.

Si vous passez un xml-datasource à la méthode parseString, les nouvelles données remplaceront les données existantes. Pour d'autres datasources, cependant, comme un in-memory-datasource, les nouvelles données seront ajoutées à celles qui sont déjà dans le datasource. Vous pourriez employer ceci pour accumuler un plus grand datasource à partir de plusieurs petits.

Quand les données sont remplacées dans un xml-datasource la manipulation spéciale est faite pour s'assurer que les données nouvellement analysées ne suppriment pas des données si elles existent déjà. Quand l'analyse se produit, les liens RDF existants dans le datasource sont en premier maintenus. Quand un nouveau lien est ajouté à partir des nouvelles données analysées, et qu'il existe déjà, le vieux lien est conservé. Si le nouveau lien n'existe pas déjà, on l'ajoute. Une fois que l'analyse est complète, les liens qui existaient dans les vieilles données mais qui n'étaient pas dans les nouvelles données sont nettoyés. Ce processus peu commun est employé pour s'assurer que les objets ressource ne soient pas supprimés et ne soient pas recréés, ce qui peux avoir des effets secondaires indésirables. Ce processus comporte l'utilisation de l'interface nsIRDFPurgeableDataSource. Cette interface interne est prévue pour être employée seulement à cette fin -- vous ne devriez pas l'employer vous-même.

La méthode parseString charge de façon synchrone. Il y a également une méthode parseAsync qui peut être employée pour analyser de façon asynchrone. Elles prennent le datasource et l'URI de base et elles retournent un objet qui met en application l'interface nsIStreamListener. Vous devrez appeler les méthodes de cette interface et lui passer le RDF. C'est un peu maladroit, ainsi vous n'emploierez probablement pas cette méthode.

Composite datasource

Le composite-datasource contient une liste d'autres datasources. Quand vous questionnez ce datasource, il questionnera chacun des datasources de la liste chacun leur tour jusqu'à ce qu'une réponse soit trouvée, qui sera alors retournée. Si vous essayez de changer le composite-datasource il appellera chacun des datasources dans sa liste jusqu'à ce qu'un d'eux accepte le changement. Pour les méthodes d'interrogation qui renvoient seulement une valeur simple, ou pour des méthodes de modification, seulement un datasource renverra des résultats ou sera changé. Une fois qu'un datasource est trouvé avec le résultat ou qu'il accepte un changement, les autres ne seront pas interrogés. Pour les méthodes d'interrogation qui renvoient normalement des valeurs multiples, toutes les valeurs possibles dans tous les datasources sont retournées. Naturellement, vous pouvez interroger et encore modifier les différents datasources séparément.

Puisque le composite-datasource contient une liste de datasources, il peut efficacement être employé comme si tous les datasources qu'il contient avaient tous été combinés dans un datasource simple. Cette assemblage s'appelle parfois l'agrégation. N'importe quel datasource peut être ajouté à un composite-datasource que ce soient des sources RDF/XML, des datasources intégrés de Mozilla et même d'autres composite-datasources.

Le datasource ajouté à un élément de XUL quand il emploie un attribut datasources est un composite-datasource, ainsi vous pouvez ajouter des datasources et les enlever à tout moment.

L'interface nsIRDFCompositeDataSource est implémentée par le composite-datasource et est employée pour ajouter et enlever des datasources. Pour ajouter un datasource, employez la méthode AddDataSource. Elle prend un argument, le datasource à ajouter. Vous devrez passer le datasource lui-même et non son URI. Notez que cette méthode ne vérifie pas l'unicité. Si le datasource est déjà inclus dans la liste, on l'ajoutera encore. Le code suivant est un exemple pour ajouter un datasource:


var rdfService = Components.classes["@mozilla.org/rdf/rdf-service;1"].
                   getService(Components.interfaces.nsIRDFService);
var compositeDS = rdfService.GetDataSource("rdf:composite-datasource");

var ds1=rdfService.GetDataSource("http://www.xulplanet.com/tutorials/xultu/animals.rdf");
var ds2=rdfService.GetDataSource("rdf:bookmarks");

compositeDS.QueryInterface(Components.interfaces.nsIRDFCompositeDataSource);
compositeDS.AddDataSource(ds1);
compositeDS.AddDataSource(ds2);

Quand vous interrogez le composite-datasource, il interrogera le premier datasource, dans ce cas-ci une source RDF/XML, suivie du second, le datasource de signets, jusqu'à ce qu'un résultat soit trouvé.

Vous pouvez enlever un datasource du composé, en appelant la méthode de RemoveDataSource. Elle prend un argument, le datasource à enlever.

La méthode GetDataSources peut être employée pour obtenir une liste des datasources employés par le composé. Cette méthode renvoie une énumération qui peut être employée pour réitérer au-dessus des datasources. Ils seront retournés dans l'ordre dans lequel ils ont été ajoutés. Dans l'exemple suivant, nous recherchons les datasources qui sont attachés à un élément XUL. Ce sont les datasources que le calibre emploierait.


var compositeDS = xulElement.database;

var list = compositeDS.GetDataSources();
while (list.hasMoreElements()){
  var ds = list.getNext();
  if (ds instanceof Components.interfaces.nsIRDFDataSource){
    ...
  }
}

L'interface des ccomposites datasources a deux propriétés. La première, allowNegativeAssertions est employé pour indiquer comment le composé manipule des affirmations négatives. Une affirmation négative est un lien RDF que le datasource indique comme faux. Si cet attribut est vrai, la valeur par défaut, le composites datasource manipulera des affirmations négatives. Si un datasource contient un lien normal vrai, et un autre contient le même lien mais faux, ils se déprogrammeront l'un l'autre. Si la propriété allowNegativeAssertions est fausse, les liens négatifs des datasources ne seront pas examinés. C'est plus rapide, ainsi vous pouvez souhaiter changer la valeur si vous savez que les datasources ne contiennent aucune affirmation négative. La plupart des datasources n'en ont pas.

La propriété coalesceDuplicateArcs indique si le composite datasource enlèvera des repliques quand il sera interrogé. Si cette valeur est vraie, et qu'une interrogattion est posée, les valeurs en double seront enlevées. C'est utile quand plusieurs datasources sont combinés ainsi vous ne devrez pas vous inquiéter du fait que les datasources multiples contiennent les mêmes données. Si cette propriété est fausse, des reproductions seront retournées, qui auront comme conséquence des interrogations légèrement plus optimales.

le local store

Le datasource rdf:local-store est inclus dans Mozilla et est employé pour contenir l'information d'état telle que la position de la fenêtre du navigateur, comment sont présentées les colonnes dans les vues d'arbre, et comment sont visibles les toolbars et les sidebars. Cette information est sauvée quand Mozilla ferme sort de la fenêtre, et est réappliquée automatiquement au contenu XUL quand la fenêtre appropriée est de nouveau ouverte. Ce processus est expliqué en détail dans la section sur des données persistantes. Pour récapituler, l'attribut persist peut être employé sur un élément XUL pour sauver des données dans le magasin local (local store) et pour les récupérer quand la fenêtre XUL sera ouverte plus tard. Bien que le magasin local contienne normalement l'information d'état de XUL, vous pouvez réellement mettre ce que vous y voulez.

Le magasin local est sauvé dans un fichier RDF/XML, 'localstore.rdf ,' dans votre annuaire de profil de Mozilla. Puisque c'est un dossier de RDF/XML, vous pouvez l'ouvrir dans un éditeur de texte et regarder l'information qu'il contient. Il est possible de modifier le fichier comme bon vous semble, bien que cela ne soit pas recommandé à moins que vous sachiez ce que vous changez.

Le local-store est toujours inclus dans une application XUL chrome quand vous employez un attribut datasources sur un élément. Le magasin local est toujours le premier datasource utilisé. Naturellement, rien ne vous empêche de l'enlever en employant plus tard la méthode RemoveDataSource des datasources composés.

Ceci signifie qu'il est possible d'ajouter de l'information au magasin local qui chevauche l'information des autres datasources utilisés dans les templates. Par exemple, si vous ajoutez la bonne information au fichier localstore.rdf, vous pourriez la faire apparaître comme si l'utilisateur avait ajouter un signet, puisque Mozilla emploie un template pour faire apparaître la liste des signets de l'utilisateur. Naturellement, le signet ne fonctionnera pas correctement puisqu'il n'est pas vraiment dans le datasource de signets.