/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * You can obtain a copy of the license at * trunk/opends/resource/legal-notices/OpenDS.LICENSE * or https://OpenDS.dev.java.net/OpenDS.LICENSE. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at * trunk/opends/resource/legal-notices/OpenDS.LICENSE. If applicable, * add the following below this CDDL HEADER, with the fields enclosed * by brackets "[]" replaced with your own identifying * information: * Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END * * * Portions Copyright 2006 Sun Microsystems, Inc. */ package org.opends.server.backends; import java.util.HashMap; import java.util.HashSet; import java.util.LinkedHashMap; import java.util.LinkedList; import org.opends.server.api.Backend; import org.opends.server.config.ConfigEntry; import org.opends.server.config.ConfigException; import org.opends.server.core.AddOperation; import org.opends.server.core.DeleteOperation; import org.opends.server.core.DirectoryException; import org.opends.server.core.DirectoryServer; import org.opends.server.core.ModifyOperation; import org.opends.server.core.ModifyDNOperation; import org.opends.server.core.SearchOperation; import org.opends.server.types.BackupConfig; import org.opends.server.types.BackupDirectory; import org.opends.server.types.Control; import org.opends.server.types.DN; import org.opends.server.types.Entry; import org.opends.server.types.LDIFExportConfig; import org.opends.server.types.LDIFImportConfig; import org.opends.server.types.RestoreConfig; import org.opends.server.types.ResultCode; import org.opends.server.types.SearchScope; import org.opends.server.types.SearchFilter; import org.opends.server.util.LDIFException; import org.opends.server.util.LDIFReader; import org.opends.server.util.LDIFWriter; import static org.opends.server.loggers.Debug.*; import static org.opends.server.messages.BackendMessages.*; import static org.opends.server.messages.MessageHandler.*; /** * This class defines a very simple backend that stores its information in * memory. This is primarily intended for testing purposes with small data * sets, as it does not have any indexing mechanism such as would be required to * achieve high performance with large data sets. It is also heavily * synchronized for simplicity at the expense of performance, rather than * providing a more fine-grained locking mechanism. *

* Entries stored in this backend are held in a * LinkedHashMap<DN,Entry> object, which ensures that the * order in which you iterate over the entries is the same as the order in which * they were inserted. By combining this with the constraint that no entry can * be added before its parent, you can ensure that iterating through the entries * will always process the parent entries before their children, which is * important for both search result processing and LDIF exports. *

* As mentioned above, no data indexing is performed, so all non-baseObject * searches require iteration through the entire data set. If this is to become * a more general-purpose backend, then additional * HashMap<ByteString,Set<DN>> objects could be used * to provide that capability. *

* There is actually one index that does get maintained within this backend, * which is a mapping between the DN of an entry and the DNs of any immediate * children of that entry. This is needed to efficiently determine whether an * entry has any children (which must not be the case for delete operations). * Modify DN operations are not currently allowed, but if such support is added * in the future, then this mapping would play an integral role in that process * as well. */ public class MemoryBackend extends Backend { /** * The fully-qualified name of this class for debugging purposes. */ private static final String CLASS_NAME = "org.opends.server.backends.MemoryBackend"; // The base DNs for this backend. private DN[] baseDNs; // The mapping between parent DNs and their immediate children. private HashMap> childDNs; // The base DNs for this backend, in a hash set. private HashSet baseDNSet; // The set of supported controls for this backend. private HashSet supportedControls; // The set of supported features for this backend. private HashSet supportedFeatures; // The mapping between entry DNs and the corresponding entries. private LinkedHashMap entryMap; /** * Creates a new backend with the provided information. All backend * implementations must implement a default constructor that use * super() to invoke this constructor. */ public MemoryBackend() { super(); assert debugConstructor(CLASS_NAME); // Perform all initialization in initializeBackend. } /** * {@inheritDoc} */ public synchronized void initializeBackend(ConfigEntry configEntry, DN[] baseDNs) throws ConfigException { assert debugEnter(CLASS_NAME, "initializeBackend", String.valueOf(configEntry), String.valueOf(baseDNs)); // We won't support anything other than exactly one base DN in this // implementation. If we were to add such support in the future, we would // likely want to separate the data for each base DN into a separate entry // map. if ((baseDNs == null) || (baseDNs.length != 1)) { int msgID = MSGID_MEMORYBACKEND_REQUIRE_EXACTLY_ONE_BASE; String message = getMessage(msgID); throw new ConfigException(msgID, message); } this.baseDNs = baseDNs; baseDNSet = new HashSet(); for (DN dn : baseDNs) { baseDNSet.add(dn); } entryMap = new LinkedHashMap(); childDNs = new HashMap>(); supportedControls = new HashSet(); supportedFeatures = new HashSet(); for (DN dn : baseDNs) { DirectoryServer.registerSuffix(dn, this); } } /** * Removes any data that may have been stored in this backend. */ public synchronized void clearMemoryBackend() { entryMap.clear(); childDNs.clear(); } /** * {@inheritDoc} */ public synchronized void finalizeBackend() { assert debugEnter(CLASS_NAME, "finalizeBackend"); clearMemoryBackend(); } /** * {@inheritDoc} */ public DN[] getBaseDNs() { assert debugEnter(CLASS_NAME, "getBaseDNs"); return baseDNs; } /** * {@inheritDoc} */ public boolean isLocal() { assert debugEnter(CLASS_NAME, "isLocal"); // For the purposes of this method, this is a local backend. return true; } /** * {@inheritDoc} */ public synchronized Entry getEntry(DN entryDN) { assert debugEnter(CLASS_NAME, "getEntry", String.valueOf(entryDN)); return entryMap.get(entryDN); } /** * {@inheritDoc} */ public synchronized boolean entryExists(DN entryDN) { assert debugEnter(CLASS_NAME, "entryExists", String.valueOf(entryDN)); return entryMap.containsKey(entryDN); } /** * {@inheritDoc} */ public synchronized void addEntry(Entry entry, AddOperation addOperation) throws DirectoryException { assert debugEnter(CLASS_NAME, "addEntry", String.valueOf(entry), String.valueOf(addOperation)); // See if the target entry already exists. If so, then fail. DN entryDN = entry.getDN(); if (entryMap.containsKey(entryDN)) { int msgID = MSGID_MEMORYBACKEND_ENTRY_ALREADY_EXISTS; String message = getMessage(msgID, String.valueOf(entryDN)); throw new DirectoryException(ResultCode.ENTRY_ALREADY_EXISTS, message, msgID); } // If the entry is one of the base DNs, then add it. if (baseDNSet.contains(entryDN)) { entryMap.put(entryDN, entry); return; } // Get the parent DN and ensure that it exists in the backend. DN parentDN = entryDN.getParent(); if (parentDN == null) { int msgID = MSGID_MEMORYBACKEND_ENTRY_DOESNT_BELONG; String message = getMessage(msgID, String.valueOf(entryDN)); throw new DirectoryException(ResultCode.NO_SUCH_OBJECT, message, msgID); } else if (! entryMap.containsKey(parentDN)) { int msgID = MSGID_MEMORYBACKEND_PARENT_DOESNT_EXIST; String message = getMessage(msgID, String.valueOf(entryDN), String.valueOf(parentDN)); throw new DirectoryException(ResultCode.NO_SUCH_OBJECT, message, msgID); } entryMap.put(entryDN, entry); HashSet children = childDNs.get(parentDN); if (children == null) { children = new HashSet(); childDNs.put(parentDN, children); } children.add(entryDN); } /** * {@inheritDoc} */ public synchronized void deleteEntry(DN entryDN, DeleteOperation deleteOperation) throws DirectoryException { assert debugEnter(CLASS_NAME, "deleteEntry", String.valueOf(entryDN), String.valueOf(deleteOperation)); // Make sure the entry exists. If not, then throw an exception. if (! entryMap.containsKey(entryDN)) { int msgID = MSGID_MEMORYBACKEND_ENTRY_DOESNT_EXIST; String message = getMessage(msgID, String.valueOf(entryDN)); throw new DirectoryException(ResultCode.NO_SUCH_OBJECT, message, msgID); } // Make sure the entry doesn't have any children. If it does, then throw an // exception. HashSet children = childDNs.get(entryDN); if ((children != null) && (! children.isEmpty())) { int msgID = MSGID_MEMORYBACKEND_CANNOT_DELETE_ENTRY_WITH_CHILDREN; String message = getMessage(msgID, String.valueOf(entryDN)); throw new DirectoryException(ResultCode.NOT_ALLOWED_ON_NONLEAF, message, msgID); } // Remove the entry from the backend. Also remove the reference to it from // its parent, if applicable. childDNs.remove(entryDN); entryMap.remove(entryDN); DN parentDN = entryDN.getParent(); if (parentDN != null) { HashSet parentsChildren = childDNs.get(parentDN); if (parentsChildren != null) { parentsChildren.remove(entryDN); if (parentsChildren.isEmpty()) { childDNs.remove(parentDN); } } } } /** * {@inheritDoc} */ public synchronized void replaceEntry(Entry entry, ModifyOperation modifyOperation) throws DirectoryException { assert debugEnter(CLASS_NAME, "replaceEntry", String.valueOf(entry), String.valueOf(modifyOperation)); // Make sure the entry exists. If not, then throw an exception. DN entryDN = entry.getDN(); if (! entryMap.containsKey(entryDN)) { int msgID = MSGID_MEMORYBACKEND_ENTRY_DOESNT_EXIST; String message = getMessage(msgID, String.valueOf(entryDN)); throw new DirectoryException(ResultCode.NO_SUCH_OBJECT, message, msgID); } // Replace the old entry with the new one. entryMap.put(entryDN, entry); } /** * {@inheritDoc} */ public synchronized void renameEntry(DN currentDN, Entry entry, ModifyDNOperation modifyDNOperation) throws DirectoryException { assert debugEnter(CLASS_NAME, "renameEntry", String.valueOf(currentDN), String.valueOf(entry), String.valueOf(modifyDNOperation)); // Make sure that the target entry exists. if (! entryMap.containsKey(currentDN)) { int msgID = MSGID_MEMORYBACKEND_ENTRY_DOESNT_EXIST; String message = getMessage(msgID, String.valueOf(currentDN)); throw new DirectoryException(ResultCode.NO_SUCH_OBJECT, message, msgID); } // Make sure that the target entry doesn't have any children. HashSet children = childDNs.get(currentDN); if (children != null) { if (children.isEmpty()) { childDNs.remove(currentDN); } else { int msgID = MSGID_MEMORYBACKEND_CANNOT_RENAME_ENRY_WITH_CHILDREN; String message = getMessage(msgID, String.valueOf(currentDN)); throw new DirectoryException(ResultCode.NOT_ALLOWED_ON_NONLEAF, message, msgID); } } // Make sure that no entry exists with the new DN. if (entryMap.containsKey(entry.getDN())) { int msgID = MSGID_MEMORYBACKEND_ENTRY_ALREADY_EXISTS; String message = getMessage(msgID, String.valueOf(entry.getDN())); throw new DirectoryException(ResultCode.NO_SUCH_OBJECT, message, msgID); } // Make sure that the new DN is in this backend. boolean matchFound = false; for (DN dn : baseDNs) { if (dn.isAncestorOf(entry.getDN())) { matchFound = true; break; } } if (! matchFound) { int msgID = MSGID_MEMORYBACKEND_CANNOT_RENAME_TO_ANOTHER_BACKEND; String message = getMessage(msgID, String.valueOf(currentDN)); throw new DirectoryException(ResultCode.UNWILLING_TO_PERFORM, message, msgID); } // Make sure that the parent of the new entry exists. DN parentDN = entry.getDN().getParent(); if ((parentDN == null) || (! entryMap.containsKey(parentDN))) { int msgID = MSGID_MEMORYBACKEND_RENAME_PARENT_DOESNT_EXIST; String message = getMessage(msgID, String.valueOf(currentDN), String.valueOf(parentDN)); throw new DirectoryException(ResultCode.UNWILLING_TO_PERFORM, message, msgID); } // Delete the current entry and add the new one. deleteEntry(currentDN, null); addEntry(entry, null); } /** * {@inheritDoc} */ public synchronized void search(SearchOperation searchOperation) throws DirectoryException { assert debugEnter(CLASS_NAME, "search", String.valueOf(searchOperation)); // Get the base DN, scope, and filter for the search. DN baseDN = searchOperation.getBaseDN(); SearchScope scope = searchOperation.getScope(); SearchFilter filter = searchOperation.getFilter(); // If it's a base-level search, then just get that entry and return it if it // matches the filter. if (scope == SearchScope.BASE_OBJECT) { Entry entry = entryMap.get(baseDN); if (entry == null) { int msgID = MSGID_MEMORYBACKEND_ENTRY_DOESNT_EXIST; String message = getMessage(msgID, String.valueOf(baseDN)); throw new DirectoryException(ResultCode.NO_SUCH_OBJECT, message, msgID); } if (filter.matchesEntry(entry)) { searchOperation.returnEntry(entry, new LinkedList()); } } else { // Walk through all entries and send the ones that match. for (Entry e : entryMap.values()) { if (e.matchesBaseAndScope(baseDN, scope) && filter.matchesEntry(e)) { searchOperation.returnEntry(e, new LinkedList()); } } } } /** * {@inheritDoc} */ public HashSet getSupportedControls() { assert debugEnter(CLASS_NAME, "getSupportedControls"); return supportedControls; } /** * {@inheritDoc} */ public boolean supportsControl(String controlOID) { assert debugEnter(CLASS_NAME, "supportsControl", String.valueOf(controlOID)); // This backend does not provide any special control support. return false; } /** * {@inheritDoc} */ public HashSet getSupportedFeatures() { assert debugEnter(CLASS_NAME, "getSupportedFeatures"); return supportedFeatures; } /** * {@inheritDoc} */ public boolean supportsFeature(String featureOID) { assert debugEnter(CLASS_NAME, "supportsFeature", String.valueOf(featureOID)); // This backend does not provide any special feature support. return false; } /** * {@inheritDoc} */ public boolean supportsLDIFExport() { assert debugEnter(CLASS_NAME, "supportsLDIFExport"); return true; } /** * {@inheritDoc} */ public synchronized void exportLDIF(ConfigEntry configEntry, DN[] baseDNs, LDIFExportConfig exportConfig) throws DirectoryException { assert debugEnter(CLASS_NAME, "exportLDIF", String.valueOf(exportConfig)); // Create the LDIF writer. LDIFWriter ldifWriter; try { ldifWriter = new LDIFWriter(exportConfig); } catch (Exception e) { assert debugException(CLASS_NAME, "exportLDIF", e); int msgID = MSGID_MEMORYBACKEND_CANNOT_CREATE_LDIF_WRITER; String message = getMessage(msgID, String.valueOf(e)); throw new DirectoryException(DirectoryServer.getServerErrorResultCode(), message, msgID, e); } // Walk through all the entries and write them to LDIF. DN entryDN = null; try { for (Entry entry : entryMap.values()) { entryDN = entry.getDN(); ldifWriter.writeEntry(entry); } } catch (Exception e) { int msgID = MSGID_MEMORYBACKEND_CANNOT_WRITE_ENTRY_TO_LDIF; String message = getMessage(msgID, String.valueOf(entryDN), String.valueOf(e)); throw new DirectoryException(DirectoryServer.getServerErrorResultCode(), message, msgID, e); } finally { try { ldifWriter.close(); } catch (Exception e) { assert debugException(CLASS_NAME, "exportLDIF", e); } } } /** * {@inheritDoc} */ public boolean supportsLDIFImport() { assert debugEnter(CLASS_NAME, "supportsLDIFImport"); return true; } /** * {@inheritDoc} */ public synchronized void importLDIF(ConfigEntry configEntry, DN[] baseDNs, LDIFImportConfig importConfig) throws DirectoryException { assert debugEnter(CLASS_NAME, "importLDIF", String.valueOf(importConfig)); clearMemoryBackend(); LDIFReader reader; try { reader = new LDIFReader(importConfig); } catch (Exception e) { int msgID = MSGID_MEMORYBACKEND_CANNOT_CREATE_LDIF_READER; String message = getMessage(msgID, String.valueOf(e)); throw new DirectoryException(DirectoryServer.getServerErrorResultCode(), message, msgID, e); } try { while (true) { Entry e = null; try { e = reader.readEntry(); if (e == null) { break; } } catch (LDIFException le) { if (! le.canContinueReading()) { int msgID = MSGID_MEMORYBACKEND_ERROR_READING_LDIF; String message = getMessage(msgID, String.valueOf(e)); throw new DirectoryException( DirectoryServer.getServerErrorResultCode(), message, msgID, le); } else { continue; } } try { addEntry(e, null); } catch (DirectoryException de) { reader.rejectLastEntry(de.getErrorMessage()); } } } catch (DirectoryException de) { throw de; } catch (Exception e) { int msgID = MSGID_MEMORYBACKEND_ERROR_DURING_IMPORT; String message = getMessage(msgID, String.valueOf(e)); throw new DirectoryException(DirectoryServer.getServerErrorResultCode(), message, msgID, e); } finally { reader.close(); } } /** * {@inheritDoc} */ public boolean supportsBackup() { assert debugEnter(CLASS_NAME, "supportsBackup"); // This backend does not provide a backup/restore mechanism. return false; } /** * {@inheritDoc} */ public boolean supportsBackup(BackupConfig backupConfig, StringBuilder unsupportedReason) { assert debugEnter(CLASS_NAME, "supportsBackup"); // This backend does not provide a backup/restore mechanism. return false; } /** * {@inheritDoc} */ public void createBackup(ConfigEntry configEntry, BackupConfig backupConfig) throws DirectoryException { assert debugEnter(CLASS_NAME, "createBackup", String.valueOf(backupConfig)); int msgID = MSGID_MEMORYBACKEND_BACKUP_RESTORE_NOT_SUPPORTED; String message = getMessage(msgID); throw new DirectoryException(ResultCode.UNWILLING_TO_PERFORM, message, msgID); } /** * {@inheritDoc} */ public void removeBackup(BackupDirectory backupDirectory, String backupID) throws DirectoryException { assert debugEnter(CLASS_NAME, "removeBackup", String.valueOf(backupDirectory), String.valueOf(backupID)); int msgID = MSGID_MEMORYBACKEND_BACKUP_RESTORE_NOT_SUPPORTED; String message = getMessage(msgID); throw new DirectoryException(ResultCode.UNWILLING_TO_PERFORM, message, msgID); } /** * {@inheritDoc} */ public boolean supportsRestore() { assert debugEnter(CLASS_NAME, "supportsRestore"); // This backend does not provide a backup/restore mechanism. return false; } /** * {@inheritDoc} */ public void restoreBackup(ConfigEntry configEntry, RestoreConfig restoreConfig) throws DirectoryException { assert debugEnter(CLASS_NAME, "restoreBackup", String.valueOf(restoreConfig)); int msgID = MSGID_MEMORYBACKEND_BACKUP_RESTORE_NOT_SUPPORTED; String message = getMessage(msgID); throw new DirectoryException(ResultCode.UNWILLING_TO_PERFORM, message, msgID); } }