aboutsummaryrefslogtreecommitdiffstats
path: root/src/net/java/sip/communicator/impl/configuration/JdbcConfigService.java
diff options
context:
space:
mode:
Diffstat (limited to 'src/net/java/sip/communicator/impl/configuration/JdbcConfigService.java')
-rw-r--r--src/net/java/sip/communicator/impl/configuration/JdbcConfigService.java910
1 files changed, 910 insertions, 0 deletions
diff --git a/src/net/java/sip/communicator/impl/configuration/JdbcConfigService.java b/src/net/java/sip/communicator/impl/configuration/JdbcConfigService.java
new file mode 100644
index 0000000..e75068b
--- /dev/null
+++ b/src/net/java/sip/communicator/impl/configuration/JdbcConfigService.java
@@ -0,0 +1,910 @@
+/*
+ * Jitsi, the OpenSource Java VoIP and Instant Messaging client.
+ *
+ * Distributable under LGPL license.
+ * See terms of license at gnu.org.
+ */
+package net.java.sip.communicator.impl.configuration;
+
+import java.beans.*;
+import java.io.*;
+import java.sql.*;
+import java.sql.Statement;
+import java.util.*;
+
+import org.jitsi.service.configuration.*;
+import org.jitsi.service.fileaccess.*;
+import org.jitsi.util.*;
+
+import com.google.common.collect.*;
+
+/**
+ * Implementation of the {@link ConfigurationService} based on JDBC.
+ *
+ * @author Ingo Bauersachs
+ */
+public final class JdbcConfigService
+ implements ConfigurationService
+{
+ /**
+ * The <tt>Logger</tt> used by this class.
+ */
+ private final Logger logger
+ = Logger.getLogger(JdbcConfigService.class);
+
+ /**
+ * Name of the file containing default properties.
+ */
+ private static final String DEFAULT_PROPS_FILE_NAME
+ = "jitsi-defaults.properties";
+
+ /**
+ * Name of the file containing overrides (possibly set by the distributor)
+ * for any of the default properties.
+ */
+ private static final String DEFAULT_OVERRIDES_PROPS_FILE_NAME
+ = "jitsi-default-overrides.properties";
+
+ /**
+ * A set of immutable properties deployed with the application during
+ * install time. The properties in this file will be impossible to override
+ * and attempts to do so will simply be ignored.
+ * @see #defaultProperties
+ */
+ private Map<String, String> immutableDefaultProperties
+ = new HashMap<String, String>();
+
+ /**
+ * A set of properties deployed with the application during install time.
+ * Contrary to the properties in {@link #immutableDefaultProperties} the
+ * ones in this map can be overridden with call to the
+ * <tt>setProperty()</tt> methods. Still, re-setting one of these properties
+ * to <tt>null</tt> would cause for its initial value to be restored.
+ */
+ private Map<String, String> defaultProperties
+ = new HashMap<String, String>();
+
+ /**
+ * Registered property change listeners that may veto a change.
+ */
+ private SetMultimap<String, ConfigVetoableChangeListener> vetoListeners
+ = HashMultimap.create();
+
+ /**
+ * Registered property change listeners.
+ */
+ private SetMultimap<String, PropertyChangeListener> listeners
+ = HashMultimap.create();
+
+ /**
+ * Connection to the JDBC database.
+ */
+ private Connection connection;
+
+ // SQL statements for queries against the database
+ private PreparedStatement selectExact;
+ private PreparedStatement selectLike;
+ private PreparedStatement selectAll;
+ private PreparedStatement insertOrUpdate;
+ private PreparedStatement delete;
+
+ /**
+ * Reference to the {@link FileAccessService}.
+ */
+ private FileAccessService fas;
+
+ /**
+ * Creates a new instance of this class.
+ * @param fas Reference to the {@link FileAccessService}.
+ * @throws Exception
+ */
+ public JdbcConfigService(FileAccessService fas) throws Exception
+ {
+ this.fas = fas;
+ File dataFile = fas.getPrivatePersistentFile(
+ "props.hsql.script",
+ FileCategory.PROFILE);
+ File oldProps = fas.getPrivatePersistentFile(
+ "sip-communicator.properties",
+ FileCategory.PROFILE);
+
+ // if the file for the current database does not exist yet but
+ // the previous properties-based file is there, migrate it
+ boolean migrate = false;
+ if (!dataFile.exists() && oldProps.exists())
+ {
+ migrate = true;
+ }
+
+ // open the connection
+ Class.forName("org.hsqldb.jdbc.JDBCDriver");
+ checkConnection();
+
+ // then do the actual migration
+ if (migrate)
+ {
+ Properties p = new Properties();
+ p.load(new FileInputStream(oldProps));
+
+ this.connection.setAutoCommit(false);
+ for (Map.Entry<Object, Object> e : p.entrySet())
+ {
+ this.setProperty(e.getKey().toString(), e.getValue(), false);
+ }
+
+ this.connection.commit();
+ this.connection.setAutoCommit(true);
+ }
+
+ // and finally load the (mandatory) system properties
+ loadDefaultProperties(DEFAULT_PROPS_FILE_NAME);
+ loadDefaultProperties(DEFAULT_OVERRIDES_PROPS_FILE_NAME);
+ }
+
+ /**
+ * Verifies that the connection to the database and all prepared statement
+ * are valid.
+ *
+ * @throws SQLException
+ */
+ private void checkConnection() throws SQLException
+ {
+ if (this.connection != null && this.connection.isValid(1))
+ {
+ return;
+ }
+
+ String filename;
+ try
+ {
+ File f = fas.getPrivatePersistentFile(
+ "props.hsql",
+ FileCategory.PROFILE);
+ filename = f.getAbsolutePath();
+ }
+ catch (Exception e)
+ {
+ throw new SQLException(e);
+ }
+
+ this.connection = DriverManager.getConnection(
+ "jdbc:hsqldb:file:"
+ + filename
+ + ";shutdown=true;hsqldb.write_delay=false;"
+ + "hsqldb.write_delay_millis=0");
+ Statement st = this.connection.createStatement();
+ st.executeUpdate(
+ "CREATE TABLE IF NOT EXISTS Props ("
+ + "k LONGVARCHAR UNIQUE, v LONGVARCHAR"
+ + ")");
+
+ this.selectExact = this.connection.prepareStatement(
+ "SELECT v FROM Props WHERE k=?");
+ this.selectLike = this.connection.prepareStatement(
+ "SELECT k, v FROM Props WHERE k LIKE ?");
+ this.selectAll = this.connection.prepareStatement(
+ "SELECT k, v FROM Props");
+ this.insertOrUpdate = this.connection.prepareStatement(
+ "MERGE INTO Props"
+ + " USING (VALUES(?,?)) AS i(k,v) ON Props.k = i.k"
+ + " WHEN MATCHED THEN UPDATE SET Props.v = i.v"
+ + " WHEN NOT MATCHED THEN INSERT (k, v) VALUES (i.k, i.v)");
+ this.delete = this.connection.prepareStatement(
+ "DELETE FROM Props WHERE k=?");
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#setProperty(java
+ * .lang.String, java.lang.Object)
+ */
+ @Override
+ public synchronized void setProperty(String propertyName, Object property)
+ {
+ this.setProperty(propertyName, property, false);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#setProperty(java
+ * .lang.String, java.lang.Object, boolean)
+ */
+ @Override
+ public synchronized void setProperty(String propertyName, Object property,
+ boolean isSystem)
+ {
+ // a property with the same name as an existing system property cannot
+ // be set, so mark it as a system property
+ if (!isSystem && System.getProperty(propertyName) != null)
+ {
+ isSystem = true;
+ }
+
+ if (isSystem)
+ {
+ if (property == null)
+ {
+ System.clearProperty(propertyName);
+ return;
+ }
+
+ System.setProperty(propertyName, property.toString());
+ }
+ else
+ {
+ if (immutableDefaultProperties.containsKey(propertyName))
+ {
+ return;
+ }
+
+ try
+ {
+ this.checkConnection();
+ Object oldValue = this.getProperty(propertyName);
+ this.fireVetoableChange(propertyName, oldValue, property);
+ if (property == null)
+ {
+ this.delete.setString(1, propertyName);
+ this.delete.execute();
+ }
+ else
+ {
+ this.insertOrUpdate.setString(1, propertyName);
+ this.insertOrUpdate.setString(2, property.toString());
+ this.insertOrUpdate.execute();
+ }
+
+ this.fireChange(propertyName, oldValue, property);
+ }
+ catch (SQLException e)
+ {
+ throw new RuntimeException(e);
+ }
+ }
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#setProperties(java
+ * .util.Map)
+ */
+ @Override
+ public synchronized void setProperties(Map<String, Object> properties)
+ {
+ try
+ {
+ checkConnection();
+ this.connection.setAutoCommit(false);
+ for (Map.Entry<String, Object> e : properties.entrySet())
+ {
+ this.setProperty(e.getKey(), e.getValue(), false);
+ }
+
+ this.connection.commit();
+ this.connection.setAutoCommit(true);
+ }
+ catch (SQLException e1)
+ {
+ throw new RuntimeException(e1);
+ }
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getProperty(java
+ * .lang.String)
+ */
+ @Override
+ public synchronized Object getProperty(String propertyName)
+ {
+ Object value = immutableDefaultProperties.get(propertyName);
+ if (value != null)
+ {
+ return value;
+ }
+
+ try
+ {
+ this.checkConnection();
+ this.selectExact.setString(1, propertyName);
+ ResultSet q = this.selectExact.executeQuery();
+ if (q.next())
+ {
+ value = q.getString(1);
+ }
+ }
+ catch (SQLException e)
+ {
+ logger.error(e);
+ throw new RuntimeException(e);
+ }
+
+ if (value != null)
+ {
+ return value;
+ }
+
+ value = defaultProperties.get(propertyName);
+ if (value != null)
+ {
+ return value;
+ }
+
+ return System.getProperty(propertyName);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#removeProperty(java
+ * .lang.String)
+ */
+ @Override
+ public synchronized void removeProperty(String propertyName)
+ {
+ //remove all properties
+ for (String child : this.getPropertyNamesByPrefix(propertyName, false))
+ {
+ removeProperty(child);
+ }
+
+ this.setProperty(propertyName, null, false);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getAllPropertyNames
+ * ()
+ */
+ @Override
+ public List<String> getAllPropertyNames()
+ {
+ List<String> data = new ArrayList<String>(
+ immutableDefaultProperties.keySet());
+ data.addAll(defaultProperties.keySet());
+ try
+ {
+ this.checkConnection();
+ ResultSet q = this.selectAll.executeQuery();
+ while (q.next())
+ {
+ data.add(q.getString(1));
+ }
+ }
+ catch (SQLException e)
+ {
+ logger.error(e);
+ throw new RuntimeException(e);
+ }
+
+ return data;
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService
+ * #getPropertyNamesByPrefix(java.lang.String, boolean)
+ */
+ @Override
+ public List<String> getPropertyNamesByPrefix(String prefix,
+ boolean exactPrefixMatch)
+ {
+ try
+ {
+ List<String> resultSet = new ArrayList<String>(50);
+ this.checkConnection();
+ this.selectLike.setString(1, prefix + "%");
+ ResultSet q = this.selectLike.executeQuery();
+ while (q.next())
+ {
+ String key = q.getString(1);
+ int ix = key.lastIndexOf('.');
+ if(ix == -1)
+ {
+ continue;
+ }
+
+ String keyPrefix = key.substring(0, ix);
+ if(exactPrefixMatch)
+ {
+ if(prefix.equals(keyPrefix))
+ {
+ resultSet.add(key);
+ }
+ }
+ else
+ {
+ if(keyPrefix.startsWith(prefix))
+ {
+ resultSet.add(key);
+ }
+ }
+ }
+
+ return resultSet;
+ }
+ catch (SQLException e)
+ {
+ throw new RuntimeException(e);
+ }
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService
+ * #getPropertyNamesBySuffix(java.lang.String)
+ */
+ @Override
+ public List<String> getPropertyNamesBySuffix(String suffix)
+ {
+ try
+ {
+ List<String> resultKeySet = new ArrayList<String>(20);
+ this.checkConnection();
+ this.selectLike.setString(1, "%" + suffix);
+ ResultSet q = this.selectLike.executeQuery();
+ while (q.next())
+ {
+ String key = q.getString(1);
+ int ix = key.lastIndexOf('.');
+ if (ix != -1 && suffix.equals(key.substring(ix + 1)))
+ resultKeySet.add(key);
+ }
+
+ return resultKeySet;
+ }
+ catch (SQLException e)
+ {
+ throw new RuntimeException(e);
+ }
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getString(java.lang
+ * .String)
+ */
+ @Override
+ public String getString(String propertyName)
+ {
+ String value = (String)this.getProperty(propertyName);
+ if (value != null)
+ {
+ value = value.trim();
+ if (value.length() == 0)
+ {
+ return null;
+ }
+ }
+
+ return value;
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getString(java.lang
+ * .String, java.lang.String)
+ */
+ @Override
+ public String getString(String propertyName, String defaultValue)
+ {
+ String value = this.getString(propertyName);
+ if (value == null)
+ {
+ return defaultValue;
+ }
+
+ return value;
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getBoolean(java.
+ * lang.String, boolean)
+ */
+ @Override
+ public boolean getBoolean(String propertyName, boolean defaultValue)
+ {
+ Object value = this.getProperty(propertyName);
+ if (value == null)
+ {
+ return defaultValue;
+ }
+
+ return Boolean.parseBoolean(value.toString());
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getInt(java.lang
+ * .String, int)
+ */
+ @Override
+ public int getInt(String propertyName, int defaultValue)
+ {
+ Object value = this.getProperty(propertyName);
+ if (value == null)
+ {
+ return defaultValue;
+ }
+
+ return Integer.parseInt(value.toString());
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getLong(java.lang
+ * .String, long)
+ */
+ @Override
+ public long getLong(String propertyName, long defaultValue)
+ {
+ Object value = this.getProperty(propertyName);
+ if (value == null)
+ {
+ return defaultValue;
+ }
+
+ return Long.parseLong(value.toString());
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see org.jitsi.service.configuration.ConfigurationService#
+ * addPropertyChangeListener(java.beans.PropertyChangeListener)
+ */
+ @Override
+ public void addPropertyChangeListener(PropertyChangeListener listener)
+ {
+ this.listeners.put(null, listener);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see org.jitsi.service.configuration.ConfigurationService#
+ * removePropertyChangeListener(java.beans.PropertyChangeListener)
+ */
+ @Override
+ public void removePropertyChangeListener(PropertyChangeListener listener)
+ {
+ this.listeners.remove(null, listener);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see org.jitsi.service.configuration.ConfigurationService#
+ * addPropertyChangeListener(java.lang.String,
+ * java.beans.PropertyChangeListener)
+ */
+ @Override
+ public void addPropertyChangeListener(String propertyName,
+ PropertyChangeListener listener)
+ {
+ this.listeners.put(propertyName, listener);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see org.jitsi.service.configuration.ConfigurationService#
+ * removePropertyChangeListener(java.lang.String,
+ * java.beans.PropertyChangeListener)
+ */
+ @Override
+ public void removePropertyChangeListener(String propertyName,
+ PropertyChangeListener listener)
+ {
+ this.listeners.remove(propertyName, listener);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see org.jitsi.service.configuration.ConfigurationService#
+ * addVetoableChangeListener
+ * (org.jitsi.service.configuration.ConfigVetoableChangeListener)
+ */
+ @Override
+ public void addVetoableChangeListener(ConfigVetoableChangeListener listener)
+ {
+ this.vetoListeners.put(null, listener);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see org.jitsi.service.configuration.ConfigurationService#
+ * removeVetoableChangeListener
+ * (org.jitsi.service.configuration.ConfigVetoableChangeListener)
+ */
+ @Override
+ public void removeVetoableChangeListener(
+ ConfigVetoableChangeListener listener)
+ {
+ this.vetoListeners.remove(null, listener);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see org.jitsi.service.configuration.ConfigurationService#
+ * addVetoableChangeListener(java.lang.String,
+ * org.jitsi.service.configuration.ConfigVetoableChangeListener)
+ */
+ @Override
+ public void addVetoableChangeListener(String propertyName,
+ ConfigVetoableChangeListener listener)
+ {
+ this.vetoListeners.put(propertyName, listener);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see org.jitsi.service.configuration.ConfigurationService#
+ * removeVetoableChangeListener(java.lang.String,
+ * org.jitsi.service.configuration.ConfigVetoableChangeListener)
+ */
+ @Override
+ public void removeVetoableChangeListener(String propertyName,
+ ConfigVetoableChangeListener listener)
+ {
+ this.vetoListeners.remove(propertyName, listener);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#storeConfiguration()
+ */
+ @Override
+ public void storeConfiguration() throws IOException
+ {
+ try
+ {
+ this.connection.close();
+ }
+ catch (SQLException e)
+ {
+ logger.error(e);
+ }
+ finally
+ {
+ this.connection = null;
+ }
+ }
+
+ /**
+ * Does nothing. The database cannot be edited from the outside.
+ */
+ @Override
+ public void reloadConfiguration() throws IOException
+ {
+ // nothing to do, the file cannot be edited outside
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#purgeStoredConfiguration
+ * ()
+ */
+ @Override
+ public void purgeStoredConfiguration()
+ {
+ try
+ {
+ this.checkConnection();
+ Statement st = this.connection.createStatement();
+ st.executeUpdate("TRUNCATE TABLE Props");
+ }
+ catch (SQLException e)
+ {
+ logger.error(e);
+ throw new RuntimeException(e);
+ }
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getScHomeDirName()
+ */
+ @Override
+ public String getScHomeDirName()
+ {
+ return System.getProperty(PNAME_SC_HOME_DIR_NAME);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getScHomeDirLocation
+ * ()
+ */
+ @Override
+ public String getScHomeDirLocation()
+ {
+ return System.getProperty(PNAME_SC_HOME_DIR_LOCATION);
+ }
+
+ /*
+ * (non-Javadoc)
+ *
+ * @see
+ * org.jitsi.service.configuration.ConfigurationService#getConfigurationFilename
+ * ()
+ */
+ @Override
+ public String getConfigurationFilename()
+ {
+ return "props.hsql.script";
+ }
+
+
+ /**
+ * Loads the specified default properties maps from the Jitsi installation
+ * directory. Typically this file is to be called for the default properties
+ * and the admin overrides.
+ *
+ * @param fileName the name of the file we need to load.
+ */
+ private void loadDefaultProperties(String fileName)
+ {
+ try
+ {
+ Properties fileProps = new Properties();
+
+ InputStream fileStream;
+ if(OSUtils.IS_ANDROID)
+ {
+ fileStream
+ = getClass().getClassLoader()
+ .getResourceAsStream(fileName);
+ }
+ else
+ {
+ fileStream = ClassLoader.getSystemResourceAsStream(fileName);
+ }
+
+ fileProps.load(fileStream);
+ fileStream.close();
+
+ // now get those properties and place them into the mutable and
+ // immutable properties maps.
+ for (Map.Entry<Object, Object> entry : fileProps.entrySet())
+ {
+ String name = (String) entry.getKey();
+ String value = (String) entry.getValue();
+
+ if ( name == null
+ || value == null
+ || name.trim().length() == 0)
+ {
+ continue;
+ }
+
+ if (name.startsWith("*"))
+ {
+ name = name.substring(1);
+
+ if(name.trim().length() == 0)
+ {
+ continue;
+ }
+
+ //it seems that we have a valid default immutable property
+ immutableDefaultProperties.put(name, value);
+
+ //in case this is an override, make sure we remove previous
+ //definitions of this property
+ defaultProperties.remove(name);
+ }
+ else
+ {
+ //this property is a regular, mutable default property.
+ defaultProperties.put(name, value);
+
+ //in case this is an override, make sure we remove previous
+ //definitions of this property
+ immutableDefaultProperties.remove(name);
+ }
+ }
+ }
+ catch (Exception ex)
+ {
+ //we can function without defaults so we are just logging those.
+ logger.info("No defaults property file loaded: " + fileName
+ + ". Not a problem.");
+
+ if(logger.isDebugEnabled())
+ logger.debug("load exception", ex);
+ }
+ }
+
+ /**
+ * Notify all listening objects about a prospective change.
+ *
+ * @param propertyName The property that is going to change.
+ * @param oldValue The previous value of the property (can be <tt>null</tt>)
+ * @param newValue The new value of the property (can be <tt>null</tt>)
+ */
+ private void fireVetoableChange(String propertyName,
+ Object oldValue, Object newValue)
+ {
+ PropertyChangeEvent evt = new PropertyChangeEvent(
+ this,
+ propertyName,
+ oldValue,
+ newValue);
+
+ for (ConfigVetoableChangeListener l : vetoListeners.get(propertyName))
+ {
+ l.vetoableChange(evt);
+ }
+
+ for (ConfigVetoableChangeListener l : vetoListeners.get(null))
+ {
+ l.vetoableChange(evt);
+ }
+ }
+
+ /**
+ * Notify all listeners that a property has changed.
+ *
+ * @param propertyName The property that has just changed.
+ * @param oldValue The previous value of the property (can be <tt>null</tt>)
+ * @param newValue The new value of the property (can be <tt>null</tt>)
+ */
+ private void fireChange(String propertyName,
+ Object oldValue, Object newValue)
+ {
+ PropertyChangeEvent evt = new PropertyChangeEvent(
+ this,
+ propertyName,
+ oldValue,
+ newValue);
+
+ for (PropertyChangeListener l : listeners.get(propertyName))
+ {
+ l.propertyChange(evt);
+ }
+
+ for (PropertyChangeListener l : listeners.get(null))
+ {
+ l.propertyChange(evt);
+ }
+ }
+}