blob: 4bec5af57d2120f9bed24e24bae52297247295f0 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*/
package org.apache.directory.studio.ldapbrowser.core.jobs;
import org.apache.directory.api.ldap.model.constants.SchemaConstants;
import org.apache.directory.api.ldap.model.entry.Entry;
import org.apache.directory.api.ldap.model.message.SearchScope;
import org.apache.directory.api.ldap.model.name.Dn;
import org.apache.directory.studio.common.core.jobs.StudioProgressMonitor;
import org.apache.directory.studio.connection.core.Connection;
import org.apache.directory.studio.connection.core.io.api.StudioSearchResultEnumeration;
import org.apache.directory.studio.connection.core.jobs.StudioConnectionBulkRunnableWithProgress;
import org.apache.directory.studio.ldapbrowser.core.BrowserCoreMessages;
import org.apache.directory.studio.ldapbrowser.core.events.BrowserConnectionUpdateEvent;
import org.apache.directory.studio.ldapbrowser.core.events.EventRegistry;
import org.apache.directory.studio.ldapbrowser.core.model.IBrowserConnection;
import org.apache.directory.studio.ldapbrowser.core.model.SearchParameter;
import org.apache.directory.studio.ldapbrowser.core.model.schema.Schema;
import org.apache.directory.studio.ldifparser.model.LdifEnumeration;
import org.apache.directory.studio.ldifparser.model.container.LdifContentRecord;
/**
* Runnable to reload the schema.
*
* @author <a href="mailto:dev@directory.apache.org">Apache Directory Project</a>
*/
public class ReloadSchemaRunnable implements StudioConnectionBulkRunnableWithProgress
{
/** The browser connection. */
private IBrowserConnection browserConnection;
/**
* Creates a new instance of ReloadSchemaRunnable.
*
* @param browserConnection the browser connection
*/
public ReloadSchemaRunnable( IBrowserConnection browserConnection )
{
this.browserConnection = browserConnection;
}
/**
* {@inheritDoc}
*/
public Connection[] getConnections()
{
return new Connection[]
{ browserConnection.getConnection() };
}
/**
* {@inheritDoc}
*/
public String getName()
{
return BrowserCoreMessages.jobs__reload_schemas_name_1;
}
/**
* {@inheritDoc}
*/
public Object[] getLockedObjects()
{
return new IBrowserConnection[]
{ browserConnection };
}
/**
* {@inheritDoc}
*/
public String getErrorMessage()
{
return BrowserCoreMessages.jobs__reload_schemas_error_1;
}
/**
* {@inheritDoc}
*/
public void run( StudioProgressMonitor monitor )
{
monitor.beginTask( " ", 3 ); //$NON-NLS-1$
monitor.reportProgress( " " ); //$NON-NLS-1$
monitor.setTaskName( BrowserCoreMessages.bind( BrowserCoreMessages.jobs__reload_schemas_task, new String[]
{ browserConnection.getConnection().getName() } ) );
monitor.worked( 1 );
// load schema
monitor.reportProgress( BrowserCoreMessages.model__loading_schema );
reloadSchema( true, browserConnection, monitor );
monitor.worked( 1 );
}
/**
* {@inheritDoc}
*/
public void runNotification( StudioProgressMonitor monitor )
{
BrowserConnectionUpdateEvent browserConnectionUpdateEvent = new BrowserConnectionUpdateEvent(
browserConnection, BrowserConnectionUpdateEvent.Detail.SCHEMA_UPDATED );
EventRegistry.fireBrowserConnectionUpdated( browserConnectionUpdateEvent, this );
}
/**
* Reloads the schema.
*
* @param forceReload true to force the reload of the schema, otherwise it would only be reloaded
* if the server-side schema is newer than the cached schema.
* @param browserConnection the browser connection
* @param monitor the progress monitor
*/
public static void reloadSchema( boolean forceReload, IBrowserConnection browserConnection,
StudioProgressMonitor monitor )
{
Dn schemaLocation = getSchemaLocation( browserConnection, monitor );
if ( schemaLocation == null )
{
monitor.reportError( BrowserCoreMessages.model__missing_schema_location );
return;
}
Schema schema = browserConnection.getSchema();
boolean mustReload = forceReload || ( schema == Schema.DEFAULT_SCHEMA )
|| mustReload( schemaLocation, browserConnection, monitor );
if ( mustReload )
{
browserConnection.setSchema( Schema.DEFAULT_SCHEMA );
try
{
SearchParameter sp = new SearchParameter();
sp.setSearchBase( schemaLocation );
sp.setFilter( Schema.SCHEMA_FILTER );
sp.setScope( SearchScope.OBJECT );
sp.setReturningAttributes( new String[]
{ SchemaConstants.OBJECT_CLASSES_AT, SchemaConstants.ATTRIBUTE_TYPES_AT,
SchemaConstants.LDAP_SYNTAXES_AT, SchemaConstants.MATCHING_RULES_AT,
SchemaConstants.MATCHING_RULE_USE_AT, SchemaConstants.CREATE_TIMESTAMP_AT,
SchemaConstants.MODIFY_TIMESTAMP_AT } );
LdifEnumeration le = ExportLdifRunnable.search( browserConnection, sp, monitor );
if ( le.hasNext() )
{
LdifContentRecord schemaRecord = ( LdifContentRecord ) le.next();
schema = new Schema();
schema.loadFromRecord( schemaRecord );
browserConnection.setSchema( schema );
}
else
{
monitor.reportError( BrowserCoreMessages.model__no_schema_information );
}
}
catch ( Exception e )
{
monitor.reportError( BrowserCoreMessages.model__error_loading_schema, e );
e.printStackTrace();
}
}
}
/**
* Checks if the schema must be reloaded
*
* @param browserConnection the browser connection
* @param monitor the progress monitor
*/
private static boolean mustReload( Dn schemaLocation, IBrowserConnection browserConnection,
StudioProgressMonitor monitor )
{
Schema schema = browserConnection.getSchema();
try
{
SearchParameter sp = new SearchParameter();
sp.setSearchBase( schemaLocation );
sp.setFilter( Schema.SCHEMA_FILTER );
sp.setScope( SearchScope.OBJECT );
sp.setReturningAttributes( new String[]
{ SchemaConstants.CREATE_TIMESTAMP_AT, SchemaConstants.MODIFY_TIMESTAMP_AT } );
StudioSearchResultEnumeration enumeration = SearchRunnable.search( browserConnection, sp, monitor );
while ( enumeration != null && enumeration.hasMore() )
{
String createTimestamp = null;
String modifyTimestamp = null;
Entry entry = enumeration.next().getEntry();
if ( entry.hasObjectClass( SchemaConstants.MODIFY_TIMESTAMP_AT ) )
{
modifyTimestamp = entry.get( SchemaConstants.MODIFY_TIMESTAMP_AT ).getString();
}
if ( entry.hasObjectClass( SchemaConstants.CREATE_TIMESTAMP_AT ) )
{
createTimestamp = entry.get( SchemaConstants.CREATE_TIMESTAMP_AT ).getString();
}
String schemaTimestamp = modifyTimestamp != null ? modifyTimestamp : createTimestamp;
String cacheTimestamp = schema.getModifyTimestamp() != null ? schema.getModifyTimestamp() : schema
.getCreateTimestamp();
if ( cacheTimestamp != null && schemaTimestamp != null
&& schemaTimestamp.compareTo( cacheTimestamp ) > 0 )
{
return true;
}
}
}
catch ( Exception e )
{
monitor.reportError( BrowserCoreMessages.model__error_loading_schema, e );
e.printStackTrace();
}
return false;
}
private static Dn getSchemaLocation( IBrowserConnection browserConnection, StudioProgressMonitor monitor )
{
try
{
SearchParameter sp = new SearchParameter();
sp.setSearchBase( new Dn() );
sp.setScope( SearchScope.OBJECT );
sp.setReturningAttributes( new String[]
{ SchemaConstants.SUBSCHEMA_SUBENTRY_AT } );
StudioSearchResultEnumeration enumeration = SearchRunnable.search( browserConnection, sp, monitor );
while ( enumeration != null && enumeration.hasMore() )
{
Entry entry = enumeration.next().getEntry();
if ( entry.containsAttribute( SchemaConstants.SUBSCHEMA_SUBENTRY_AT ) )
{
String value = entry.get( SchemaConstants.SUBSCHEMA_SUBENTRY_AT ).getString();
if ( Dn.isValid( value ) )
{
Dn dn = new Dn( value );
return dn;
}
}
}
}
catch ( Exception e )
{
monitor.reportError( BrowserCoreMessages.model__error_loading_schema, e );
return null;
}
return null;
}
}