mirror of
https://github.com/cookiengineer/audacity
synced 2025-05-04 17:49:45 +02:00
967 lines
29 KiB
C
967 lines
29 KiB
C
/* -*- Mode: c; c-basic-offset: 2 -*-
|
|
*
|
|
* rdf_storage_trees.c - RDF Storage in memory using balanced trees
|
|
*
|
|
* Copyright (C) 2000-2008, David Beckett http://www.dajobe.org/
|
|
* Copyright (C) 2000-2004, University of Bristol, UK http://www.bristol.ac.uk/
|
|
*
|
|
* This package is Free Software and part of Redland http://librdf.org/
|
|
*
|
|
* It is licensed under the following three licenses as alternatives:
|
|
* 1. GNU Lesser General Public License (LGPL) V2.1 or any newer version
|
|
* 2. GNU General Public License (GPL) V2 or any newer version
|
|
* 3. Apache License, V2.0 or any newer version
|
|
*
|
|
* You may not use this file except in compliance with at least one of
|
|
* the above three licenses.
|
|
*
|
|
* See LICENSE.html or LICENSE.txt at the top of this package for the
|
|
* complete terms and further detail along with the license texts for
|
|
* the licenses in COPYING.LIB, COPYING and LICENSE-2.0.txt respectively.
|
|
*
|
|
*
|
|
*/
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <rdf_config.h>
|
|
#endif
|
|
|
|
#ifdef WIN32
|
|
#include <win32_rdf_config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#ifdef HAVE_STDLIB_H
|
|
#include <stdlib.h> /* for abort() as used in errors */
|
|
#endif
|
|
#include <sys/types.h>
|
|
|
|
#include <redland.h>
|
|
#include "rdf_avltree_internal.h"
|
|
|
|
/* Not yet fully implemented (namely iteration) */
|
|
/*#define RDF_STORAGE_TREES_WITH_CONTEXTS 1*/
|
|
|
|
typedef struct
|
|
{
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
librdf_node* context;
|
|
#endif
|
|
librdf_avltree* spo_tree; /* Always present */
|
|
librdf_avltree* sop_tree; /* Optional */
|
|
librdf_avltree* ops_tree; /* Optional */
|
|
librdf_avltree* pso_tree; /* Optional */
|
|
} librdf_storage_trees_graph;
|
|
|
|
typedef struct
|
|
{
|
|
librdf_storage_trees_graph* graph; /* Statements without a context */
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
librdf_avltree* contexts; /* Tree of librdf_storage_trees_graph */
|
|
#endif
|
|
int index_sop;
|
|
int index_ops;
|
|
int index_pso;
|
|
} librdf_storage_trees_context;
|
|
|
|
/* prototypes for local functions */
|
|
static int librdf_storage_trees_init(librdf_storage* storage, const char *name, librdf_hash* options);
|
|
static int librdf_storage_trees_open(librdf_storage* storage, librdf_model* model);
|
|
static int librdf_storage_trees_close(librdf_storage* storage);
|
|
static int librdf_storage_trees_size(librdf_storage* storage);
|
|
static int librdf_storage_trees_add_statement(librdf_storage* storage, librdf_statement* statement);
|
|
static int librdf_storage_trees_add_statements(librdf_storage* storage, librdf_stream* statement_stream);
|
|
static int librdf_storage_trees_remove_statement(librdf_storage* storage, librdf_statement* statement);
|
|
static int librdf_storage_trees_remove_statement_internal(librdf_storage_trees_graph* graph, librdf_statement* statement);
|
|
static int librdf_storage_trees_contains_statement(librdf_storage* storage, librdf_statement* statement);
|
|
static librdf_stream* librdf_storage_trees_serialise(librdf_storage* storage);
|
|
static librdf_stream* librdf_storage_trees_find_statements(librdf_storage* storage, librdf_statement* statement);
|
|
|
|
/* graph functions */
|
|
static librdf_storage_trees_graph* librdf_storage_trees_graph_new(librdf_storage* storage, librdf_node* context);
|
|
static void librdf_storage_trees_graph_free(void* data);
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
static int librdf_storage_trees_graph_compare(const void* data1, const void* data2);
|
|
#endif
|
|
|
|
/* serialising implementing functions */
|
|
static int librdf_storage_trees_serialise_end_of_stream(void* context);
|
|
static int librdf_storage_trees_serialise_next_statement(void* context);
|
|
static void* librdf_storage_trees_serialise_get_statement(void* context, int flags);
|
|
static void librdf_storage_trees_serialise_finished(void* context);
|
|
|
|
/* context functions */
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
static int librdf_storage_trees_context_add_statement(librdf_storage* storage, librdf_node* context_node, librdf_statement* statement);
|
|
static int librdf_storage_trees_context_remove_statement(librdf_storage* storage, librdf_node* context_node, librdf_statement* statement);
|
|
static librdf_stream* librdf_storage_trees_context_serialise(librdf_storage* storage, librdf_node* context_node);
|
|
#endif
|
|
|
|
/* statement tree functions */
|
|
static int librdf_statement_compare_spo(const void* data1, const void* data2);
|
|
static int librdf_statement_compare_sop(const void* data1, const void* data2);
|
|
static int librdf_statement_compare_ops(const void* data1, const void* data2);
|
|
static int librdf_statement_compare_pso(const void* data1, const void* data2);
|
|
static void librdf_storage_trees_avl_free(void* data);
|
|
|
|
|
|
static void librdf_storage_trees_register_factory(librdf_storage_factory *factory);
|
|
|
|
|
|
|
|
/* functions implementing storage api */
|
|
static int
|
|
librdf_storage_trees_init(librdf_storage* storage, const char *name,
|
|
librdf_hash* options)
|
|
{
|
|
librdf_storage_trees_context *context=(librdf_storage_trees_context*)storage->context;
|
|
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
/* Support contexts if option given */
|
|
if (librdf_hash_get_as_boolean(options, "contexts") > 0) {
|
|
context->contexts=librdf_new_avltree(librdf_storage_trees_graph_compare,
|
|
librdf_storage_trees_graph_free);
|
|
} else {
|
|
context->contexts=NULL;
|
|
}
|
|
#endif
|
|
|
|
const int index_spo_option = librdf_hash_get_as_boolean(options, "index-spo") > 0;
|
|
const int index_sop_option = librdf_hash_get_as_boolean(options, "index-sop") > 0;
|
|
const int index_ops_option = librdf_hash_get_as_boolean(options, "index-ops") > 0;
|
|
const int index_pso_option = librdf_hash_get_as_boolean(options, "index-pso") > 0;
|
|
|
|
/* No indexing options given, index all by default */
|
|
if (!index_spo_option && !index_sop_option && !index_ops_option && !index_pso_option) {
|
|
context->index_sop=1;
|
|
context->index_ops=1;
|
|
context->index_pso=1;
|
|
} else {
|
|
/* spo is always indexed, option just exists so user can
|
|
* specifically /only/ index spo */
|
|
context->index_sop=index_sop_option;
|
|
context->index_ops=index_ops_option;
|
|
context->index_pso=index_pso_option;
|
|
}
|
|
|
|
context->graph = librdf_storage_trees_graph_new(storage, NULL);
|
|
|
|
/* no more options, might as well free them now */
|
|
if(options)
|
|
librdf_free_hash(options);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void
|
|
librdf_storage_trees_terminate(librdf_storage* storage)
|
|
{
|
|
/* nop */
|
|
}
|
|
|
|
|
|
static int
|
|
librdf_storage_trees_open(librdf_storage* storage, librdf_model* model)
|
|
{
|
|
/* nop */
|
|
return 0;
|
|
}
|
|
|
|
|
|
/**
|
|
* librdf_storage_trees_close:
|
|
* @storage: the storage
|
|
*
|
|
* .
|
|
*
|
|
* Close the storage, and free all content since there is no persistance.
|
|
*
|
|
* Return value: non 0 on failure
|
|
**/
|
|
static int
|
|
librdf_storage_trees_close(librdf_storage* storage)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
|
|
librdf_storage_trees_graph_free(context->graph);
|
|
context->graph=NULL;
|
|
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
librdf_free_avltree(context->contexts);
|
|
context->contexts=NULL;
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int
|
|
librdf_storage_trees_size(librdf_storage* storage)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
|
|
return librdf_avltree_size(context->graph->spo_tree);
|
|
}
|
|
|
|
|
|
static int
|
|
librdf_storage_trees_add_statement_internal(librdf_storage* storage,
|
|
librdf_storage_trees_graph* graph,
|
|
librdf_statement* statement)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
int status = 0;
|
|
|
|
/* copy statement (store single copy in all trees) */
|
|
statement = librdf_new_statement_from_statement(statement);
|
|
|
|
/* spo_tree owns statement */
|
|
status = librdf_avltree_add(graph->spo_tree, statement);
|
|
if (status) /* LIBRDF_AVLTREE_ENOMEM or LIBRDF_AVLTREE_EXISTS */
|
|
return status;
|
|
|
|
/* others have null deleters */
|
|
/* (XXX: corrupt model if insertions fail) */
|
|
|
|
if (context->index_sop)
|
|
librdf_avltree_add(graph->sop_tree, statement);
|
|
|
|
if (context->index_ops)
|
|
librdf_avltree_add(graph->ops_tree, statement);
|
|
|
|
if (context->index_pso)
|
|
librdf_avltree_add(graph->pso_tree, statement);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
/**
|
|
* librdf_storage_trees_add_statement:
|
|
* @storage: #librdf_storage object
|
|
* @context_node: #librdf_node object
|
|
* @statement: #librdf_statement statement to add
|
|
*
|
|
* Add a statement (with no context) to the storage.
|
|
*
|
|
* Return value: non 0 on failure (negative if error, positive if statement
|
|
* already exists).
|
|
**/
|
|
static int
|
|
librdf_storage_trees_add_statement(librdf_storage* storage,
|
|
librdf_statement* statement)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
return librdf_storage_trees_add_statement_internal(storage, context->graph, statement);
|
|
}
|
|
|
|
|
|
static int
|
|
librdf_storage_trees_add_statements(librdf_storage* storage,
|
|
librdf_stream* statement_stream)
|
|
{
|
|
int status=0;
|
|
|
|
for(; !librdf_stream_end(statement_stream); librdf_stream_next(statement_stream)) {
|
|
librdf_statement* statement=librdf_stream_get_object(statement_stream);
|
|
|
|
if (statement) {
|
|
status=librdf_storage_trees_add_statement(storage, statement);
|
|
if (status)
|
|
break;
|
|
} else {
|
|
status=1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
static int
|
|
librdf_storage_trees_remove_statement_internal(librdf_storage_trees_graph* graph,
|
|
librdf_statement* statement)
|
|
{
|
|
if (graph->sop_tree)
|
|
librdf_avltree_delete(graph->sop_tree, statement);
|
|
|
|
if (graph->ops_tree)
|
|
librdf_avltree_delete(graph->ops_tree, statement);
|
|
|
|
if (graph->pso_tree)
|
|
librdf_avltree_delete(graph->pso_tree, statement);
|
|
|
|
return librdf_avltree_delete(graph->spo_tree, statement);
|
|
}
|
|
|
|
|
|
/**
|
|
* librdf_storage_trees_remove_statement:
|
|
* @storage: #librdf_storage object
|
|
* @context_node: #librdf_node object
|
|
* @statement: #librdf_statement statement to remove
|
|
*
|
|
* Remove a statement (without context) from the storage.
|
|
*
|
|
* Return value: non 0 on failure
|
|
**/
|
|
static int
|
|
librdf_storage_trees_remove_statement(librdf_storage* storage,
|
|
librdf_statement* statement)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
|
|
return librdf_storage_trees_remove_statement_internal(context->graph, statement);
|
|
}
|
|
|
|
static int
|
|
librdf_storage_trees_contains_statement(librdf_storage* storage, librdf_statement* statement)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
|
|
return (librdf_avltree_search(context->graph->spo_tree, statement) != NULL);
|
|
}
|
|
|
|
|
|
typedef struct {
|
|
librdf_storage *storage;
|
|
librdf_iterator *iterator;
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
librdf_node *context_node;
|
|
#endif
|
|
} librdf_storage_trees_serialise_stream_context;
|
|
|
|
|
|
static librdf_stream*
|
|
librdf_storage_trees_serialise_range(librdf_storage* storage, librdf_statement* range)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
librdf_storage_trees_serialise_stream_context* scontext;
|
|
librdf_stream* stream;
|
|
int filter = 0;
|
|
|
|
scontext=(librdf_storage_trees_serialise_stream_context*)LIBRDF_CALLOC(librdf_storage_trees_serialise_stream_context, 1, sizeof(librdf_storage_trees_serialise_stream_context));
|
|
if(!scontext)
|
|
return NULL;
|
|
|
|
scontext->iterator = NULL;
|
|
|
|
/* ?s ?p ?o */
|
|
if (!range || (!range->subject && !range->predicate && !range->object)) {
|
|
scontext->iterator=librdf_avltree_get_iterator_start(storage->world, context->graph->spo_tree,
|
|
NULL, NULL);
|
|
if (range) {
|
|
librdf_free_statement(range);
|
|
range=NULL;
|
|
}
|
|
/* s ?p o */
|
|
} else if (range->subject && !range->predicate && range->object) {
|
|
if (context->index_sop)
|
|
scontext->iterator=librdf_avltree_get_iterator_start(storage->world, context->graph->sop_tree,
|
|
range, librdf_storage_trees_avl_free);
|
|
else
|
|
filter=1;
|
|
/* s _ _ */
|
|
} else if (range->subject) {
|
|
scontext->iterator=librdf_avltree_get_iterator_start(storage->world, context->graph->spo_tree,
|
|
range, librdf_storage_trees_avl_free);
|
|
/* ?s _ o */
|
|
} else if (range->object) {
|
|
if (context->index_ops)
|
|
scontext->iterator=librdf_avltree_get_iterator_start(storage->world, context->graph->ops_tree,
|
|
range, librdf_storage_trees_avl_free);
|
|
else
|
|
filter=1;
|
|
/* ?s p ?o */
|
|
} else { /* range->predicate != NULL */
|
|
if (context->index_pso)
|
|
scontext->iterator=librdf_avltree_get_iterator_start(storage->world, context->graph->pso_tree,
|
|
range, librdf_storage_trees_avl_free);
|
|
else
|
|
filter=1;
|
|
}
|
|
|
|
/* If filter is set, we're missing the required index.
|
|
* Iterate over the entire model and filter the stream.
|
|
* (With a fully indexed store, this will never happen) */
|
|
if (filter) {
|
|
scontext->iterator=librdf_avltree_get_iterator_start(storage->world, context->graph->spo_tree,
|
|
range, librdf_storage_trees_avl_free);
|
|
}
|
|
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
scontext->context_node=NULL;
|
|
#endif
|
|
|
|
if(!scontext->iterator) {
|
|
LIBRDF_FREE(librdf_storage_trees_serialise_stream_context, scontext);
|
|
return librdf_new_empty_stream(storage->world);
|
|
}
|
|
|
|
scontext->storage=storage;
|
|
librdf_storage_add_reference(scontext->storage);
|
|
|
|
stream=librdf_new_stream(storage->world,
|
|
(void*)scontext,
|
|
&librdf_storage_trees_serialise_end_of_stream,
|
|
&librdf_storage_trees_serialise_next_statement,
|
|
&librdf_storage_trees_serialise_get_statement,
|
|
&librdf_storage_trees_serialise_finished);
|
|
|
|
if(!stream) {
|
|
librdf_storage_trees_serialise_finished((void*)scontext);
|
|
return NULL;
|
|
}
|
|
|
|
if(filter) {
|
|
if(librdf_stream_add_map(stream, &librdf_stream_statement_find_map, NULL, (void*)range)) {
|
|
/* error - stream_add_map failed */
|
|
librdf_free_stream(stream);
|
|
stream=NULL;
|
|
}
|
|
}
|
|
|
|
return stream;
|
|
}
|
|
|
|
|
|
static librdf_stream*
|
|
librdf_storage_trees_serialise(librdf_storage* storage)
|
|
{
|
|
return librdf_storage_trees_serialise_range(storage, NULL);
|
|
}
|
|
|
|
|
|
static int
|
|
librdf_storage_trees_serialise_end_of_stream(void* context)
|
|
{
|
|
librdf_storage_trees_serialise_stream_context* scontext=(librdf_storage_trees_serialise_stream_context*)context;
|
|
|
|
return librdf_iterator_end(scontext->iterator);
|
|
}
|
|
|
|
static int
|
|
librdf_storage_trees_serialise_next_statement(void* context)
|
|
{
|
|
librdf_storage_trees_serialise_stream_context* scontext=(librdf_storage_trees_serialise_stream_context*)context;
|
|
|
|
return librdf_iterator_next(scontext->iterator);
|
|
}
|
|
|
|
|
|
static void*
|
|
librdf_storage_trees_serialise_get_statement(void* context, int flags)
|
|
{
|
|
librdf_storage_trees_serialise_stream_context* scontext=(librdf_storage_trees_serialise_stream_context*)context;
|
|
|
|
switch(flags) {
|
|
case LIBRDF_ITERATOR_GET_METHOD_GET_OBJECT:
|
|
return (librdf_statement*)librdf_iterator_get_object(scontext->iterator);
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
case LIBRDF_ITERATOR_GET_METHOD_GET_CONTEXT:
|
|
return scontext->context_node;
|
|
#endif
|
|
default:
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
librdf_storage_trees_serialise_finished(void* context)
|
|
{
|
|
librdf_storage_trees_serialise_stream_context* scontext=(librdf_storage_trees_serialise_stream_context*)context;
|
|
|
|
if(scontext->iterator)
|
|
librdf_free_iterator(scontext->iterator);
|
|
|
|
if(scontext->storage)
|
|
librdf_storage_remove_reference(scontext->storage);
|
|
|
|
LIBRDF_FREE(librdf_storage_trees_serialise_stream_context, scontext);
|
|
}
|
|
|
|
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
/**
|
|
* librdf_storage_trees_context_add_statement:
|
|
* @storage: #librdf_storage object
|
|
* @context_node: #librdf_node object
|
|
* @statement: #librdf_statement statement to add
|
|
*
|
|
* Add a statement to a storage context.
|
|
*
|
|
* Return value: non 0 on failure
|
|
**/
|
|
static int
|
|
librdf_storage_trees_context_add_statement(librdf_storage* storage,
|
|
librdf_node* context_node,
|
|
librdf_statement* statement)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
|
|
librdf_storage_trees_graph* key=librdf_storage_trees_graph_new(storage, context_node);
|
|
librdf_storage_trees_graph* graph=(librdf_storage_trees_graph*)
|
|
librdf_avltree_search(context->contexts, key);
|
|
|
|
if(graph) {
|
|
librdf_storage_trees_graph_free(key);
|
|
} else {
|
|
librdf_avltree_add(context->contexts, key);
|
|
graph=key;
|
|
}
|
|
|
|
return librdf_storage_trees_add_statement_internal(storage, graph, statement);
|
|
}
|
|
|
|
|
|
/**
|
|
* librdf_storage_trees_context_remove_statement:
|
|
* @storage: #librdf_storage object
|
|
* @context_node: #librdf_node object
|
|
* @statement: #librdf_statement statement to remove
|
|
*
|
|
* Remove a statement from a storage context.
|
|
*
|
|
* Return value: non 0 on failure
|
|
**/
|
|
static int
|
|
librdf_storage_trees_context_remove_statement(librdf_storage* storage,
|
|
librdf_node* context_node,
|
|
librdf_statement* statement)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
librdf_storage_trees_graph* key=librdf_storage_trees_graph_new(storage, context_node);
|
|
librdf_storage_trees_graph* graph=(librdf_storage_trees_graph*)
|
|
librdf_avltree_search(context->contexts, &key);
|
|
librdf_storage_trees_graph_free(key);
|
|
if (graph) {
|
|
return librdf_storage_trees_remove_statement_internal(graph, statement);
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* librdf_storage_trees_context_serialise:
|
|
* @storage: #librdf_storage object
|
|
* @context_node: #librdf_node object
|
|
*
|
|
* List all statements in a storage context.
|
|
*
|
|
* Return value: #librdf_stream of statements or NULL on failure or context is empty
|
|
**/
|
|
static librdf_stream*
|
|
librdf_storage_trees_context_serialise(librdf_storage* storage,
|
|
librdf_node* context_node)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/**
|
|
* librdf_storage_trees_context_get_contexts:
|
|
* @storage: #librdf_storage object
|
|
*
|
|
* List all context nodes in a storage.
|
|
*
|
|
* Return value: #librdf_iterator of context_nodes or NULL on failure or no contexts
|
|
**/
|
|
static librdf_iterator*
|
|
librdf_storage_trees_get_contexts(librdf_storage* storage)
|
|
{
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
|
|
/**
|
|
* librdf_storage_trees_find_statements:
|
|
* @storage: the storage
|
|
* @statement: the statement to match
|
|
*
|
|
* .
|
|
*
|
|
* Return a stream of statements matching the given statement (or
|
|
* all statements if NULL). Parts (subject, predicate, object) of the
|
|
* statement can be empty in which case any statement part will match that.
|
|
* Uses #librdf_statement_match to do the matching.
|
|
*
|
|
* Return value: a #librdf_stream or NULL on failure
|
|
**/
|
|
static librdf_stream*
|
|
librdf_storage_trees_find_statements(librdf_storage* storage, librdf_statement* statement)
|
|
{
|
|
librdf_stream* stream;
|
|
|
|
librdf_statement* range=librdf_new_statement_from_statement(statement);
|
|
if(!range)
|
|
return NULL;
|
|
|
|
stream=librdf_storage_trees_serialise_range(storage, range);
|
|
|
|
return stream;
|
|
}
|
|
|
|
/* statement tree functions */
|
|
|
|
static int
|
|
librdf_storage_trees_node_compare(librdf_node* n1, librdf_node* n2)
|
|
{
|
|
if (n1 == n2) {
|
|
return 0;
|
|
} else if (n1->type != n2->type) {
|
|
return n2->type - n1->type;
|
|
} else {
|
|
switch (n1->type) {
|
|
case LIBRDF_NODE_TYPE_RESOURCE:
|
|
return librdf_uri_compare(n1->value.resource.uri, n2->value.resource.uri);
|
|
case LIBRDF_NODE_TYPE_LITERAL:
|
|
return strcmp((char*)n1->value.literal.string, (char*)n2->value.literal.string);
|
|
case LIBRDF_NODE_TYPE_BLANK:
|
|
return strcmp((char*)n1->value.blank.identifier, (char*)n2->value.blank.identifier);
|
|
case LIBRDF_NODE_TYPE_UNKNOWN:
|
|
default:
|
|
return (char*)n2-(char*)n1; /* ? */
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* Compare two statements in (s, p, o) order.
|
|
* NULL fields act as wildcards. */
|
|
static int
|
|
librdf_statement_compare_spo(const void* data1, const void* data2)
|
|
{
|
|
librdf_statement* a = (librdf_statement*)data1;
|
|
librdf_statement* b = (librdf_statement*)data2;
|
|
int cmp = 0;
|
|
|
|
/* Subject */
|
|
if (a->subject == NULL || b->subject == NULL)
|
|
return 0; /* wildcard subject match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->subject, b->subject);
|
|
|
|
if (cmp != 0)
|
|
return cmp;
|
|
|
|
/* Predicate */
|
|
if (a->predicate == NULL || b->predicate == NULL)
|
|
return 0; /* wildcard predicate match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->predicate, b->predicate);
|
|
|
|
if (cmp != 0)
|
|
return cmp;
|
|
|
|
/* Object */
|
|
if (a->object == NULL || b->object == NULL)
|
|
return 0; /* wildcard object match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->object, b->object);
|
|
|
|
return cmp;
|
|
}
|
|
|
|
|
|
/* Compare two statements in (o, s, p) order.
|
|
* NULL fields act as wildcards. */
|
|
static int
|
|
librdf_statement_compare_sop(const void* data1, const void* data2)
|
|
{
|
|
librdf_statement* a = (librdf_statement*)data1;
|
|
librdf_statement* b = (librdf_statement*)data2;
|
|
int cmp = 0;
|
|
|
|
/* Subject */
|
|
if (a->subject == NULL || b->subject == NULL)
|
|
return 0; /* wildcard subject match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->subject, b->subject);
|
|
|
|
if (cmp != 0)
|
|
return cmp;
|
|
|
|
/* Object */
|
|
if (a->object == NULL || b->object == NULL)
|
|
return 0; /* wildcard object match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->object, b->object);
|
|
|
|
if (cmp != 0)
|
|
return cmp;
|
|
|
|
/* Predicate */
|
|
if (a->predicate == NULL || b->predicate == NULL)
|
|
return 0; /* wildcard predicate match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->predicate, b->predicate);
|
|
|
|
return cmp;
|
|
}
|
|
|
|
|
|
/* Compare two statements in (o, p, s) order.
|
|
* NULL fields act as wildcards. */
|
|
static int
|
|
librdf_statement_compare_ops(const void* data1, const void* data2)
|
|
{
|
|
librdf_statement* a = (librdf_statement*)data1;
|
|
librdf_statement* b = (librdf_statement*)data2;
|
|
int cmp = 0;
|
|
|
|
/* Object */
|
|
if (a->object == NULL || b->object == NULL)
|
|
return 0; /* wildcard object match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->object, b->object);
|
|
|
|
if (cmp != 0)
|
|
return cmp;
|
|
|
|
/* Predicate */
|
|
if (a->predicate == NULL || b->predicate == NULL)
|
|
return 0; /* wildcard predicate match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->predicate, b->predicate);
|
|
|
|
if (cmp != 0)
|
|
return cmp;
|
|
|
|
/* Subject */
|
|
if (a->subject == NULL || b->subject == NULL)
|
|
return 0; /* wildcard subject match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->subject, b->subject);
|
|
|
|
return cmp;
|
|
}
|
|
|
|
|
|
/* Compare two statements in (p, s, o) order.
|
|
* NULL fields act as wildcards. */
|
|
static int
|
|
librdf_statement_compare_pso(const void* data1, const void* data2)
|
|
{
|
|
librdf_statement* a = (librdf_statement*)data1;
|
|
librdf_statement* b = (librdf_statement*)data2;
|
|
int cmp = 0;
|
|
|
|
/* Predicate */
|
|
if (a->predicate == NULL || b->predicate == NULL)
|
|
return 0; /* wildcard predicate match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->predicate, b->predicate);
|
|
|
|
if (cmp != 0)
|
|
return cmp;
|
|
|
|
/* Subject */
|
|
if (a->subject == NULL || b->subject == NULL)
|
|
return 0; /* wildcard subject match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->subject, b->subject);
|
|
|
|
if (cmp != 0)
|
|
return cmp;
|
|
|
|
/* Object */
|
|
if (a->object == NULL || b->object == NULL)
|
|
return 0; /* wildcard object match */
|
|
else
|
|
cmp = librdf_storage_trees_node_compare(a->object, b->object);
|
|
|
|
return cmp;
|
|
}
|
|
|
|
|
|
static void
|
|
librdf_storage_trees_avl_free(void* data)
|
|
{
|
|
librdf_statement* stmnt=(librdf_statement*)data;
|
|
librdf_free_statement(stmnt);
|
|
}
|
|
|
|
|
|
/* graph functions */
|
|
|
|
static librdf_storage_trees_graph*
|
|
librdf_storage_trees_graph_new(librdf_storage* storage, librdf_node* context_node)
|
|
{
|
|
librdf_storage_trees_context* context=(librdf_storage_trees_context*)storage->context;
|
|
librdf_storage_trees_graph* graph=(librdf_storage_trees_graph*)LIBRDF_MALLOC(
|
|
librdf_storage_trees_graph, sizeof(librdf_storage_trees_graph));
|
|
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
graph->context=(context_node ? librdf_new_node_from_node(context_node) : NULL);
|
|
#endif
|
|
|
|
/* Always create SPO index */
|
|
graph->spo_tree=librdf_new_avltree(librdf_statement_compare_spo, librdf_storage_trees_avl_free);
|
|
if(!graph->spo_tree) {
|
|
LIBRDF_FREE(librdf_storage_trees_graph, graph);
|
|
return NULL;
|
|
}
|
|
|
|
if(context->index_sop)
|
|
graph->sop_tree=librdf_new_avltree(librdf_statement_compare_sop, NULL);
|
|
else
|
|
graph->sop_tree=NULL;
|
|
|
|
if(context->index_ops)
|
|
graph->ops_tree=librdf_new_avltree(librdf_statement_compare_ops, NULL);
|
|
else
|
|
graph->ops_tree=NULL;
|
|
|
|
if(context->index_pso)
|
|
graph->pso_tree=librdf_new_avltree(librdf_statement_compare_pso, NULL);
|
|
else
|
|
graph->pso_tree=NULL;
|
|
|
|
return graph;
|
|
}
|
|
|
|
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
static int
|
|
librdf_storage_trees_graph_compare(const void* data1, const void* data2)
|
|
{
|
|
librdf_storage_trees_graph* a = (librdf_storage_trees_graph*)data1;
|
|
librdf_storage_trees_graph* b = (librdf_storage_trees_graph*)data2;
|
|
return librdf_storage_trees_node_compare(a->context, b->context);
|
|
}
|
|
#endif
|
|
|
|
|
|
static void
|
|
librdf_storage_trees_graph_free(void* data)
|
|
{
|
|
librdf_storage_trees_graph* graph = (librdf_storage_trees_graph*)data;
|
|
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
librdf_free_node(graph->context);
|
|
#endif
|
|
|
|
/* Extra index trees have null deleters (statements are shared) */
|
|
if (graph->sop_tree)
|
|
librdf_free_avltree(graph->sop_tree);
|
|
if (graph->ops_tree)
|
|
librdf_free_avltree(graph->ops_tree);
|
|
if (graph->pso_tree)
|
|
librdf_free_avltree(graph->pso_tree);
|
|
|
|
/* Free spo tree and statements */
|
|
librdf_free_avltree(graph->spo_tree);
|
|
|
|
graph->spo_tree=NULL;
|
|
graph->sop_tree=NULL;
|
|
graph->ops_tree=NULL;
|
|
graph->pso_tree=NULL;
|
|
|
|
LIBRDF_FREE(librdf_storage_trees_graph, graph);
|
|
}
|
|
|
|
|
|
/**
|
|
* librdf_storage_trees_get_feature:
|
|
* @storage: #librdf_storage object
|
|
* @feature: #librdf_uri feature property
|
|
*
|
|
* Get the value of a storage feature.
|
|
*
|
|
* Return value: #librdf_node feature value or NULL if no such feature
|
|
* exists or the value is empty.
|
|
**/
|
|
static librdf_node*
|
|
librdf_storage_trees_get_feature(librdf_storage* storage, librdf_uri* feature)
|
|
{
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
librdf_storage_trees_context* scontext=(librdf_storage_trees_context*)storage->context;
|
|
unsigned char *uri_string;
|
|
|
|
if(!feature)
|
|
return NULL;
|
|
|
|
uri_string=librdf_uri_as_string(feature);
|
|
if(!uri_string)
|
|
return NULL;
|
|
|
|
if(!strcmp((const char*)uri_string, LIBRDF_MODEL_FEATURE_CONTEXTS)) {
|
|
unsigned char value[2];
|
|
|
|
sprintf((char*)value, "%d", (scontext->contexts != NULL));
|
|
return librdf_new_node_from_typed_literal(storage->world,
|
|
value, NULL, NULL);
|
|
}
|
|
#endif
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* local function to register tree storage functions */
|
|
|
|
static void
|
|
librdf_storage_trees_register_factory(librdf_storage_factory *factory)
|
|
{
|
|
factory->context_length = sizeof(librdf_storage_trees_context);
|
|
|
|
factory->init = librdf_storage_trees_init;
|
|
factory->clone = NULL;
|
|
factory->terminate = librdf_storage_trees_terminate;
|
|
factory->open = librdf_storage_trees_open;
|
|
factory->close = librdf_storage_trees_close;
|
|
factory->size = librdf_storage_trees_size;
|
|
factory->add_statement = librdf_storage_trees_add_statement;
|
|
factory->add_statements = librdf_storage_trees_add_statements;
|
|
factory->remove_statement = librdf_storage_trees_remove_statement;
|
|
factory->contains_statement = librdf_storage_trees_contains_statement;
|
|
factory->serialise = librdf_storage_trees_serialise;
|
|
|
|
factory->find_statements = librdf_storage_trees_find_statements;
|
|
/* These could be implemented, but only if all indexes are available.
|
|
* If they returned NULL if the indexes weren't available,
|
|
* librdf_storage_find_statements would break, unfortunately.
|
|
* Since these are exposed by model methods, the storage interface
|
|
* needs to be fixed so these can be exposed but find_statements
|
|
* still work */
|
|
factory->find_sources = NULL;
|
|
factory->find_arcs = NULL;
|
|
factory->find_targets = NULL;
|
|
|
|
#ifdef RDF_STORAGE_TREES_WITH_CONTEXTS
|
|
factory->context_add_statement = librdf_storage_trees_context_add_statement;
|
|
factory->context_remove_statement = librdf_storage_trees_context_remove_statement;
|
|
factory->context_serialise = librdf_storage_trees_context_serialise;
|
|
factory->get_contexts = librdf_storage_trees_get_contexts;
|
|
#else
|
|
factory->context_add_statement = NULL;
|
|
factory->context_remove_statement = NULL;
|
|
factory->context_serialise = NULL;
|
|
factory->get_contexts = NULL;
|
|
#endif
|
|
|
|
factory->sync = NULL;
|
|
factory->get_feature = librdf_storage_trees_get_feature;
|
|
}
|
|
|
|
|
|
/**
|
|
* librdf_init_storage_trees:
|
|
* @world: world object
|
|
*
|
|
* INTERNAL - initialise the storage_trees module.
|
|
**/
|
|
void
|
|
librdf_init_storage_trees(librdf_world *world)
|
|
{
|
|
librdf_storage_register_factory(world, "trees", "Balanced trees",
|
|
&librdf_storage_trees_register_factory);
|
|
}
|
|
|