diff options
Diffstat (limited to 'src/net/java/sip/communicator/impl/configuration/JdbcConfigService.java')
-rw-r--r-- | src/net/java/sip/communicator/impl/configuration/JdbcConfigService.java | 910 |
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); + } + } +} |