/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- * * The contents of this file are subject to the Netscape Public License * Version 1.0 (the "NPL"); you may not use this file except in * compliance with the NPL. You may obtain a copy of the NPL at * http://www.mozilla.org/NPL/ * * Software distributed under the NPL is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL * for the specific language governing rights and limitations under the * NPL. * * The Initial Developer of this code under the NPL is Netscape * Communications Corporation. Portions created by Netscape are * Copyright (C) 1998 Netscape Communications Corporation. All Rights * Reserved. */ // 39204897 #include #include #include #include "rdf-int.h" #include "gs.h" typedef struct _TrieNodeStruct { char c; struct _TrieNodeStruct* child; struct _TrieNodeStruct* next; struct _TrieTargetStruct* targets; } TrieNodeStruct; typedef TrieNodeStruct* TNS; typedef struct _TrieTargetStruct { RDF_Resource label; RDF_Resource target; struct _TrieTargetStruct* next; RDFT db; } TrieTargetStruct; typedef TrieTargetStruct* TTS; static TNS gRootNode = 0; void addTarget (RDFT db, TNS node, RDF_Resource label, RDF_Resource targetNode) { TTS target = (TTS) fgetMem(sizeof(TrieTargetStruct)); target->next = node->targets; node->targets = target; target->label = label; target->target = targetNode; target->db = db; } TNS findChildOfChar (TNS node, char c) { TNS ch = node->child; char c1 = tolower(c); while (ch) { if (c1 == ch->c) return ch; ch = ch->next; } return 0; } TNS findChildOfString (TNS node, char* str) { size_t size = strlen(str); size_t n = 0; while (n < size) { char c = str[n++]; node = findChildOfChar(node, c); if (!node) return 0; } return node; } void RDFGS_AddSearchIndex (RDFT db, char* string, RDF_Resource label, RDF_Resource target) { size_t size = strlen(string); size_t n = 0; TNS prev, next; if (!gRootNode) gRootNode = (TNS) getMem(sizeof(TrieNodeStruct)); prev = gRootNode; next = 0; while (n < size) { char c = string[n++]; if (!wsCharp(c) && (c != '/')) { next = (TNS) findChildOfChar(prev, c); if (!next) { next = (TNS)fgetMem(sizeof(TrieNodeStruct)); next->next = prev->child; prev->child = next; next->c = tolower(c); } prev = next; } else if (next) { addTarget(db, next, label, target); prev = gRootNode; next = 0; } } if (next) addTarget(db, next, label, target); } void countChildren (TNS node, size_t *n) { TNS ch; if (node->targets) *n++; ch = node->child; while (ch) { countChildren(ch, n); ch = ch->next; } } void fillUpChildren (RDF_Cursor c, TNS node) { TNS ch; if (node->targets) *((TNS*)c->pdata1 + c->count++) = node; ch = node->child; while (ch) { fillUpChildren(c, ch); ch = ch->next; } } RDF_Cursor RDFGS_Search (RDFT db, char* searchString, RDF_Resource label) { RDF_Cursor c = (RDF_Cursor) getMem(sizeof(RDF_CursorStruct)); size_t n = 0; c->searchString = searchString; c->s = label; c->db = db; c->pdata = findChildOfString(gRootNode, searchString); countChildren((TNS)c->pdata, &n); if (n > 0) { c->count = 0; c->pdata1 = getMem(n+1); fillUpChildren(c, (TNS)c->pdata); c->count = 0; } if (c->pdata) c->pdata = ((TNS)c->pdata)->targets; return c; } void RDFGS_DisposeCursor (RDF_Cursor c) { freeMem(c); } RDF_Resource RDFGS_NextValue (RDF_Cursor c) { TNS currentTNS; TTS currentTTS = (TTS) c->pdata; if (!currentTTS) { return 0; } else { while (currentTTS) { if ((!c->s) || (c->s == currentTTS->label)) { RDF_Resource ans =currentTTS->target; currentTTS = c->pdata = currentTTS->next; return ans; } c->pdata = currentTTS = currentTTS->next; } } currentTNS = ((TNS*)c->pdata1)[c->count++]; if (!currentTNS) return 0; c->pdata = currentTNS->targets; return RDFGS_NextValue(c); }