All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.tentackle.dbms.prefs.DbPreferencesFactory Maven / Gradle / Ivy

There is a newer version: 21.16.1.0
Show newest version
/*
 * Tentackle - a framework for java desktop applications
 * http://www.tentackle.org
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

package org.tentackle.dbms.prefs;

import org.tentackle.common.Service;
import org.tentackle.dbms.Db;
import org.tentackle.dbms.prefs.DbPreferencesOperation.RefreshInfo;
import org.tentackle.log.Logger;
import org.tentackle.log.LoggerFactory;
import org.tentackle.misc.IdSerialTuple;
import org.tentackle.prefs.PersistedPreferencesFactory;
import org.tentackle.session.ModificationEvent;
import org.tentackle.session.ModificationEventDetail;
import org.tentackle.session.ModificationListenerAdapter;
import org.tentackle.session.ModificationTracker;
import org.tentackle.session.PersistenceException;
import org.tentackle.session.Session;
import org.tentackle.session.SessionUtilities;

import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.TreeSet;
import java.util.prefs.BackingStoreException;
import java.util.prefs.NodeChangeEvent;
import java.util.prefs.NodeChangeListener;
import java.util.prefs.PreferenceChangeEvent;
import java.util.prefs.PreferenceChangeListener;
import java.util.prefs.Preferences;


/**
 * Repository and factory for database backed preferences.
 * 

* The DbPreferencesFactory implements {@link PersistedPreferencesFactory} which in turn * extends {@link java.util.prefs.PreferencesFactory}. As such, it can be used as a drop-in * replacement for the default JRE preferences factory. By default, however, it is available * to the application only via {@link PersistedPreferencesFactory#getInstance()}, which is the * preferred way for tentackle applications to use preferences. * * @author harald */ @Service(PersistedPreferencesFactory.class) public class DbPreferencesFactory implements PersistedPreferencesFactory { /** * Gets the factory singleton. * * @return the factory */ public static DbPreferencesFactory getInstance() { return (DbPreferencesFactory) PersistedPreferencesFactory.getInstance(); } /** * The logger for this class. */ private static final Logger LOGGER = LoggerFactory.getLogger(DbPreferencesFactory.class); /** * Modification tracker name. */ private static final String TRACKER_NAME = "preferences"; /** * Listeners are managed by the repository, not the nodes. * This allows un- and reloading the nodes without losing the listeners. *

* Notice that when a node is removed, their listeners will not be removed. * So, if the node is added again, the listeners are still there. * Listeners have to be removed by the application explicitly. */ private static class ListenerKey { private final String user; // "" = system private final String path; private ListenerKey(DbPreferences node) { user = node.isUserNode() ? node.getUser() : ""; path = node.absolutePath(); } @Override public int hashCode() { int hash = 7; hash = 31 * hash + Objects.hashCode(this.user); hash = 31 * hash + Objects.hashCode(this.path); return hash; } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } final ListenerKey other = (ListenerKey) obj; if (!Objects.equals(this.user, other.user)) { return false; } return Objects.equals(this.path, other.path); } } /** * The root node with all nodes and keys loaded. */ private static class Root { private static long lastSerial; // last serial private final DbPreferences node; // the root node holding the whole tree private long serial; // the serial version of the whole tree /** * Creates a root node. * * @param node the preferences node */ private Root(DbPreferences node) { this.node = node; countUpdate(); } private synchronized void countUpdate() { serial = ++lastSerial; } } private boolean autoSync; // autosync flag private boolean readOnly; // readonly flag private boolean systemOnly; // systemonly flagc private final Map roots; // loaded roots, key is username ("" for system) private final Map> nodeListeners; // listeners for added/removed nodes private final Map> prefListeners; // listeners for changed keys private long nodeTableSerial; // highest tableSerial of _all_ DbPreferencesNodes or remote repo private long keyTableSerial; // highest tableSerial of _all_ DbPreferencesKeys /** * Creates the factory. */ public DbPreferencesFactory() { roots = new HashMap<>(); nodeListeners = new HashMap<>(); prefListeners = new HashMap<>(); autoSync = true; Db db = requestDb(false, false); try { if (db.isRemote()) { nodeTableSerial = ModificationTracker.getInstance().getSerial(TRACKER_NAME); keyTableSerial = 0; ModificationTracker.getInstance().addModificationListener( new ModificationListenerAdapter(TRACKER_NAME) { @Override public void dataChanged(ModificationEvent ev) { if (isAutoSync()) { updateRepository((Db) ev.getSession(), ev.getSerial(), 0); } } }); } else { nodeTableSerial = new DbPreferencesNode(db).getModificationCount(); keyTableSerial = new DbPreferencesKey(db).getModificationCount(); ModificationTracker.getInstance().addModificationListener( new ModificationListenerAdapter(DbPreferencesNode.CLASSVARIABLES.tableName, DbPreferencesKey.CLASSVARIABLES.tableName) { @Override public void dataChanged(ModificationEvent ev) { long nodeMaxSerial = 0; long keyMaxSerial = 0; for (ModificationEventDetail detail : ev.getDetails()) { if (DbPreferencesNode.CLASSVARIABLES.tableName.equals(detail.getName())) { nodeMaxSerial = detail.getSerial(); } else if (DbPreferencesKey.CLASSVARIABLES.tableName.equals(detail.getName())) { keyMaxSerial = detail.getSerial(); } } if (isAutoSync()) { updateRepository((Db) ev.getSession(), nodeMaxSerial, keyMaxSerial); } } }); } } finally { releaseDb(db); } } @Override public synchronized void invalidate() { roots.clear(); } @Override public boolean isAutoSync() { return autoSync; } @Override public void setAutoSync(boolean autoSync) { this.autoSync = autoSync; } @Override public boolean isReadOnly() { return readOnly; } @Override public void setReadOnly(boolean readOnly) { this.readOnly = readOnly; } @Override public boolean isSystemOnly() { return systemOnly; } @Override public void setSystemOnly(boolean systemOnly) { this.systemOnly = systemOnly; } @Override public DbPreferences getSystemRoot() { return getRootTree(null, null); } @Override public DbPreferences getUserRoot() { if (isSystemOnly()) { return getSystemRoot(); } return getRootTree(null, getUserName()); } @Override public DbPreferences userNodeForPackage(Class c) { return getUserRoot().node(nodeName(c)); } @Override public DbPreferences systemNodeForPackage(Class c) { return getSystemRoot().node(nodeName(c)); } @Override public Preferences systemRoot() { return getSystemRoot(); } @Override public Preferences userRoot() { return getUserRoot(); } /** * Gets the current serial version of given user- or system root. * * @param user the username, null of "" if system * @return the serial, 0 if no such root loaded */ public long getSerial(String user) { String key = user == null ? "" : user; Root root = roots.get(key); return root == null ? 0 : root.serial; } /** * Adds a preferences changed listener. *

* Notice that when a node is removed, their listeners will not be removed. * So, if the node is added again, the listeners are still there. * Listeners have to be removed by the application explicitly. * * @param node the node * @param pcl the listener */ public synchronized void addPreferenceChangeListener(DbPreferences node, PreferenceChangeListener pcl) { ListenerKey lk = new ListenerKey(node); Set listeners = prefListeners.computeIfAbsent(lk, k -> new HashSet<>()); listeners.add(pcl); } /** * Removes a preferences changed listener. * * @param node the node * @param pcl the listener */ public synchronized void removePreferenceChangeListener(DbPreferences node, PreferenceChangeListener pcl) { ListenerKey lk = new ListenerKey(node); Set listeners = prefListeners.get(lk); if (listeners != null) { listeners.remove(pcl); if (listeners.isEmpty()) { prefListeners.remove(lk); } } } /** * Adds a node listener. *

* Notice that when a node is removed, their listeners will not be removed. * So, if the node is added again, the listeners are still there. * Listeners have to be removed by the application explicitly. * * @param node the node * @param ncl the listener */ public synchronized void addNodeChangeListener(DbPreferences node, NodeChangeListener ncl) { ListenerKey lk = new ListenerKey(node); Set listeners = nodeListeners.computeIfAbsent(lk, k -> new HashSet<>()); listeners.add(ncl); } /** * Removes a node listener. * * @param node the node * @param ncl the listener */ public synchronized void removeNodeChangeListener(DbPreferences node, NodeChangeListener ncl) { ListenerKey lk = new ListenerKey(node); Set listeners = nodeListeners.get(lk); if (listeners != null) { listeners.remove(ncl); if (listeners.isEmpty()) { nodeListeners.remove(lk); } } } /** * Invoke listeners when a node is added or removed. * * @param node the added or removed node */ public void nodeChanged(DbPreferences node) { DbPreferences parent = node.parent(); if (parent != null) { Set listeners = nodeListeners.get(new ListenerKey(parent)); if (listeners != null) { NodeChangeEvent ev = new NodeChangeEvent(node.parent(), node); for (NodeChangeListener listener: listeners) { if (node.getPersistentNode().isRemoved()) { listener.childRemoved(ev); } else { listener.childAdded(ev); } } } } if (node.isRootNode() && node.getPersistentNode().isRemoved()) { removeRoot(node); } } /** * Invoke listeners when a key is added, updated or removed. * * @param node the node holding the key * @param key the changed key */ public void keyChanged(DbPreferences node, DbPreferencesKey key) { Set listeners = prefListeners.get(new ListenerKey(node)); if (listeners != null) { PreferenceChangeEvent ev = new PreferenceChangeEvent(node, key.getKey(), key.isRemoved() ? null : key.getValue()); for (PreferenceChangeListener listener: listeners) { listener.preferenceChange(ev); } } } /** * Flushes a node and all of its subnodes to the backing store. * * @param node the preferences node * @param sync true if load modifications from storage before flush * @throws BackingStoreException if failed */ public synchronized void flush(DbPreferences node, boolean sync) throws BackingStoreException { Db db = requestDb(node.isUserNode(), true); try { boolean updated = db.transaction(() -> { try { return new DbPreferencesOperation(db).flush(node, sync, false); } catch (RuntimeException rex) { // remove the root node from the repo (forces reload on next access) removeRoot(node); throw rex; } }); if (updated) { LOGGER.warning("nodes where updated or deleted by another JVM"); } } catch (RuntimeException rex) { throw new BackingStoreException(rex); } finally { releaseDb(db); } } /** * Removes a root node of given node from the repository. * * @param node the root node or one of its subnodes */ protected synchronized void removeRoot(DbPreferences node) { String user = node.getUser(); if (user == null) { user = ""; } roots.remove(user); } /** * Updates the repository.
* Invoked whenever the data in prefnode or prefkey has changed. * * @param db the session * @param nodeMaxSerial the new table serial for prefnodes * @param keyMaxSerial the nore table serial for prefkeys */ protected synchronized void updateRepository(Db db, long nodeMaxSerial, long keyMaxSerial) { if (nodeMaxSerial != nodeTableSerial || keyMaxSerial != keyTableSerial) { if (db.isRemote()) { // rebuild all roots if serials don't match with the ones in the remote repository for (Root root: roots.values()) { rebuildRoot(db, root); } } else { // get tableserial/id-pairs boolean someRemoved = false; List nodeExpireSet = null; List keyExpireSet = null; if (nodeMaxSerial > 0) { nodeExpireSet = new DbPreferencesNode(db).getExpiredTableSerials(nodeTableSerial, nodeMaxSerial); if (SessionUtilities.getInstance().isSomeRemoved(nodeTableSerial, nodeExpireSet, nodeMaxSerial)) { someRemoved = true; } } if (!someRemoved && keyMaxSerial > 0) { keyExpireSet = new DbPreferencesKey(db).getExpiredTableSerials(keyTableSerial, keyMaxSerial); if (SessionUtilities.getInstance().isSomeRemoved(keyTableSerial, keyExpireSet, keyMaxSerial)) { someRemoved = true; } } Set usersModified = new HashSet<>(); // roots affected, "" is system // TreeSet to sort by id (because childs are always added _after_ their parents, see flushImpl) Set changedNodes = new TreeSet<>(); Set changedKeys = new TreeSet<>(); Map idNodeMap = new HashMap<>(); // related nodes map by ID if (!someRemoved) { if (nodeExpireSet != null) { // figure out roots to update List prefNodes = new DbPreferencesNode(db).selectObjectsWithExpiredTableSerials(nodeTableSerial); if (prefNodes.size() == nodeExpireSet.size()) { for (DbPreferencesNode prefNode : prefNodes) { changedNodes.add(prefNode); idNodeMap.put(prefNode.getId(), prefNode); String user = prefNode.getUser(); usersModified.add(user == null ? "" : user); } } else { // removed in the meantime??? someRemoved = true; } } if (!someRemoved && keyExpireSet != null) { // update keys List prefKeys = new DbPreferencesKey(db).selectObjectsWithExpiredTableSerials(keyTableSerial); if (prefKeys.size() == keyExpireSet.size()) { for (DbPreferencesKey prefKey : prefKeys) { changedKeys.add(prefKey); DbPreferencesNode prefNode = idNodeMap.get(prefKey.getNodeId()); if (prefNode == null) { prefNode = new DbPreferencesNode(db).selectObject(prefKey.getNodeId()); if (prefNode != null) { idNodeMap.put(prefNode.getId(), prefNode); } } if (prefNode != null) { changedNodes.add(prefNode); // add even if node itself wasn't modified bec. of listeners String user = prefNode.getUser(); usersModified.add(user == null ? "" : user); } } } else { someRemoved = true; } } } if (someRemoved) { // rebuild all roots for (Root root: roots.values()) { rebuildRoot(db, root); } } else { // only added or updated nodes or keys: update roots for (String user: usersModified) { Root root = roots.get(user); if (root != null) { updateRoot(root, changedNodes, changedKeys); } } } } nodeTableSerial = nodeMaxSerial; keyTableSerial = keyMaxSerial; if (!db.isRemote()) { ModificationTracker.getInstance().countModification(db, TRACKER_NAME); } } } /** * Rebuilds the given root from scratch.
* Reloads all nodes and keys. * * @param db the session * @param root the root node */ private void rebuildRoot(Db db, Root root) { // load all nodes and keys for this root (ordered by id, i.e. creation) RefreshInfo info = new DbPreferencesOperation(db).loadRefreshInfo(root.node, db.isRemote() ? root.serial : 0); if (info != null) { Collection prefNodes = info.prefNodes; Collection prefKeys = info.prefKeys; // to speed up: build maps of ID:node and ID:key Map idNodeMap = new HashMap<>(); Map pathNodeMap = new HashMap<>(); Map idKeyMap = new HashMap<>(); addNodeToMaps(root.node, idNodeMap, pathNodeMap, idKeyMap); // add or update nodes for (DbPreferencesNode prefNode : prefNodes) { DbPreferences node = pathNodeMap.get(prefNode.getName()); if (node == null) { // node is new DbPreferences parent = idNodeMap.get(prefNode.getParentId()); // must exist bec. nodes are sorted by id if (parent == null) { // must exist, because prefNodes are sorted by id! LOGGER.severe("parent ID={0} missing for node ID={1}", prefNode.getParentId(), prefNode.getId()); removeRoot(root.node); // force reload next access } else { node = new DbPreferences(parent, prefNode, null, null); parent.addChild(node); idNodeMap.put(prefNode.getId(), node); nodeChanged(node); } } else { // just id,serial or tableserial changed if (node.updatePersistentNode(prefNode)) { nodeChanged(node); idNodeMap.put(prefNode.getId(), node); // in case ID has changed (node was new) } } } // add or update keys for (DbPreferencesKey prefKey: prefKeys) { DbPreferences node = idNodeMap.get(prefKey.getNodeId()); // must exist now! if (node == null) { LOGGER.severe("node ID={0} missing for key ID={1}", prefKey.getNodeId(), prefKey.getId()); removeRoot(root.node); // force reload next access } else { boolean updated; DbPreferencesKey key = node.getPersistentKey(prefKey.getKey()); if (key == null) { node.putPersistentKey(prefKey); idKeyMap.put(prefKey.getId(), prefKey); updated = true; } else { updated = node.updatePersistentKey(prefKey); } if (updated) { keyChanged(node, prefKey); } } } // remove keys Set persistedKeys = new HashSet<>(prefKeys); for (DbPreferencesKey prefKey: idKeyMap.values()) { if (!prefKey.isNew() && !persistedKeys.contains(prefKey)) { // if not created in this JVM DbPreferences node = idNodeMap.get(prefKey.getNodeId()); if (node != null) { prefKey = node.removePersistentKey(prefKey.getKey()); if (prefKey != null) { keyChanged(node, prefKey); } } } } // remove nodes Set persistedNodes = new HashSet<>(prefNodes); for (DbPreferences node: idNodeMap.values()) { if (!node.getPersistentNode().isNew() && !persistedNodes.contains(node.getPersistentNode())) { // if not created in this JVM DbPreferences parent = node.parent(); if (parent != null) { node = parent.removeChild(node.name()); if (node != null) { nodeChanged(node); } } } } if (db.isRemote()) { root.serial = info.serial; } else { root.countUpdate(); } } } /** * Updates a root node. * * @param root the root preferences node * @param changedNodes the changed nodes * @param changedKeys the changes keys */ private void updateRoot(Root root, Set changedNodes, Set changedKeys) { // to speed up: build maps of ID:node and ID:key Map idNodeMap = new HashMap<>(); Map pathNodeMap = new HashMap<>(); addNodeToMaps(root.node, idNodeMap, pathNodeMap); long rootId = root.node.getPersistentNode().getId(); if (rootId == 0) { // special handling if this is the first node for this user at all: // in this case the changedNodes contain the root id for (DbPreferencesNode node: changedNodes) { if (Objects.equals(root.node.getUser(), node.getUser())) { rootId = node.getRootNodeId(); break; } } } // add or update nodes for (DbPreferencesNode prefNode: changedNodes) { if (prefNode.getRootNodeId() == rootId) { // only nodes belonging to this root DbPreferences node = pathNodeMap.get(prefNode.getName()); if (node == null) { // add node DbPreferences parent = idNodeMap.get(prefNode.getParentId()); // must exist bec. nodes are sorted by id if (parent != null) { node = new DbPreferences(parent, prefNode, null, null); parent.addChild(node); idNodeMap.put(prefNode.getId(), node); nodeChanged(node); } else { LOGGER.warning("parent node ID=" + prefNode.getParentId() + " vanished for node " + prefNode); } } else { if (node.updatePersistentNode(prefNode)) { nodeChanged(node); idNodeMap.put(prefNode.getId(), node); // in case ID changed } } } } // now that all nodes are added: add or update the keys for (DbPreferencesKey prefKey: changedKeys) { if (prefKey.getRootNodeId() == rootId) { DbPreferences node = idNodeMap.get(prefKey.getNodeId()); // must exist now if (node != null) { if (node.updatePersistentKey(prefKey)) { keyChanged(node, prefKey); } } else { LOGGER.warning("node ID=" + prefKey.getNodeId() + " valinished for key " + prefKey); } } } root.countUpdate(); } /** * Requests for a session to work with.
* By default, the thread-local session is used. * If there is no default session, the mod tracker's session is used. *

* In preparation for replication layers such as poolkeeper, * a requested session must be released after use. * * @param forUser true if need a user session (no modtracker fallback) * @param forWrite true if the session is used for writing (flush), else read * @return the session */ protected Db requestDb(boolean forUser, boolean forWrite) { Db db = (Db) Session.getCurrentSession(); if (db == null) { if (forUser) { throw new PersistenceException("no thread local session for " + Thread.currentThread()); } // no thread-local session: use the one from the mod tracker db = (Db) ModificationTracker.getInstance().getSession(); if (db == null) { throw new PersistenceException("no session for " + ModificationTracker.getInstance()); } } return db; } /** * Releases the session after use. * * @param db the session */ protected void releaseDb(Db db) { // default does nothing } /** * Gets the current user name. * * @return the username */ protected String getUserName() { return requestDb(true, false).getSessionInfo().getUserName(); } /** * Gets the root node for given user and all of its subnodes and keys at once. * * @param db the oprional db, null if request a db temporarily * @param user the username, null or empty if system user * @return the root node, never null */ public synchronized DbPreferences getRootTree(Db db, String user) { String key = user == null ? "" : user; Root root = roots.get(key); if (root == null) { boolean needRelease = false; if (db == null) { db = requestDb(true, false); needRelease = true; } try { root = new Root(new DbPreferencesOperation(db).loadRootTree(user, !db.isRemote())); roots.put(key, root); } finally { if (needRelease) { releaseDb(db); } } } return root.node; } /** * Returns the absolute path name of the node corresponding to the package of the specified object. * * @param clazz the class * @return the path name * @throws IllegalArgumentException if the package has node preferences node associated with it. */ private String nodeName(Class clazz) { if (clazz.isArray()) { throw new IllegalArgumentException("Arrays have no associated preferences node"); } String className = clazz.getName(); int pkgEndIndex = className.lastIndexOf('.'); if (pkgEndIndex < 0) { return "/"; } String packageName = className.substring(0, pkgEndIndex); return "/" + packageName.replace('.', '/'); } /** * Adds all subnodes and keys of a node to given maps. * * @param node the node * @param idNodeMap the nodes mapped by ID * @param pathNodeMap the nodes mapped by path * @param idKeyMap the keys mapped by ID */ private void addNodeToMaps(DbPreferences node, Map idNodeMap, Map pathNodeMap, Map idKeyMap) { if (!node.getPersistentNode().isVirgin()) { idNodeMap.put(node.getPersistentNode().getId(), node); } pathNodeMap.put(node.absolutePath(), node); for (DbPreferencesKey key: node.getKeys()) { if (!key.isVirgin()) { idKeyMap.put(key.getId(), key); } } for (DbPreferences child: node.getChilds()) { addNodeToMaps(child, idNodeMap, pathNodeMap, idKeyMap); } } /** * Adds all subnodes of a node to given maps. * * @param node the node * @param idNodeMap the nodes mapped by ID * @param pathNodeMap the nodes mapped by path */ private void addNodeToMaps(DbPreferences node, Map idNodeMap, Map pathNodeMap) { if (!node.getPersistentNode().isVirgin()) { idNodeMap.put(node.getPersistentNode().getId(), node); } pathNodeMap.put(node.absolutePath(), node); for (DbPreferences child: node.getChilds()) { addNodeToMaps(child, idNodeMap, pathNodeMap); } } /** * Checks all preferences and tries to fix if possible.
* The method is provided since {@link DbPreferencesNode} and {@link DbPreferencesKey} provide no referential integrity * via database-enforced foreign keys, but only at application level.
* Invoke this method, whenever the database has been modified manually via SQL. * * @param db the session (must be local) */ public void checkAllPreferences(Db db) { db.assertNotRemote(); // must be a direct JDBC link db.transaction(() -> { Map nodeMap = new HashMap<>(); // map ID:Node Set processedNodes = new HashSet<>(); // already processed nodes Collection allNodes = new DbPreferencesNode(db).selectAllObjects(); Collection allKeys = new DbPreferencesKey(db).selectAllObjects(); LOGGER.info("processing {0} nodes and {1} keys", allNodes.size(), allKeys.size()); for (DbPreferencesNode node : allNodes) { nodeMap.put(node.getId(), node); } processPreferences(processedNodes, nodeMap, allKeys); allNodes.removeAll(processedNodes); for (DbPreferencesNode node: allNodes) { LOGGER.severe("orphan node found: {0}", node); } for (DbPreferencesKey key: allKeys) { LOGGER.severe("oprhan key found: {0}", key); } return null; }); } private void processPreferences(Set processedNodes, Map nodeMap, Collection allKeys) { // check root nodes for (DbPreferencesNode node : nodeMap.values()) { if (node.getParentId() == 0 && node.getRootNodeId() != node.getId()) { node.setRootNodeId(node.getId()); node.saveObject(); LOGGER.info("fixed rootNodeId of root-node {0} to {1}", node, node.getRootNodeId()); } } // process parent chain to root for (DbPreferencesNode node: nodeMap.values()) { if (node.getParentId() != 0) { DbPreferencesNode root = findRoot(nodeMap, node); if (root == null) { LOGGER.severe("missing root for node {0}", node.getId()); } else { // go down all subnodes and keys and set the rootNodeId if missing processNode(processedNodes, nodeMap.values(), allKeys, root); } } } } private void processNode(Set processedNodes, Collection allNodes, Collection allKeys, DbPreferencesNode node) { if (!processedNodes.contains(node)) { for (DbPreferencesKey key: findKeys(allKeys, node)) { if (key.getRootNodeId() != node.getRootNodeId()) { key.setRootNodeId(node.getRootNodeId()); key.saveObject(); LOGGER.info("fixed rootNodeId of key {0} to {1}", key, key.getRootNodeId()); } } for (DbPreferencesNode child: findChilds(allNodes, node)) { if (child.getRootNodeId() != node.getRootNodeId()) { child.setRootNodeId(node.getRootNodeId()); child.saveObject(); LOGGER.info("fixed rootNodeId of node {0} to {1}", child, child.getRootNodeId()); } processNode(processedNodes, allNodes, allKeys, child); } processedNodes.add(node); } } private DbPreferencesNode findRoot(Map nodeMap, DbPreferencesNode node) { while (node != null && node.getParentId() != 0) { node = nodeMap.get(node.getParentId()); } return node; } private Collection findChilds(Collection allNodes, DbPreferencesNode node) { Collection childs = new ArrayList<>(); for (DbPreferencesNode child: allNodes) { if (child.getParentId() == node.getId()) { childs.add(child); } } return childs; } private Collection findKeys(Collection allKeys, DbPreferencesNode node) { Collection keys = new ArrayList<>(); for (Iterator iter = allKeys.iterator(); iter.hasNext(); ) { DbPreferencesKey key = iter.next(); if (key.getNodeId() == node.getId()) { keys.add(key); iter.remove(); } } return keys; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy