org.ops4j.pax.runner.Run Maven / Gradle / Ivy
/*
* Copyright 2006 Niclas Hedhman.
* Copyright 2007 Alin Dreghiciu.
* Copyright 2007 David Leangen.
*
* Licensed 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.ops4j.pax.runner;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.commons.logging.LogLevel;
import org.apache.felix.framework.Logger;
import org.apache.felix.framework.ServiceRegistry;
import org.apache.felix.framework.util.EventDispatcher;
import org.ops4j.io.FileUtils;
import org.ops4j.lang.NullArgumentException;
import org.ops4j.pax.runner.commons.Info;
import org.ops4j.pax.runner.osgi.RunnerBundle;
import org.ops4j.pax.runner.osgi.RunnerBundleContext;
import org.ops4j.pax.runner.osgi.RunnerStartLevel;
import org.ops4j.pax.runner.platform.*;
import org.ops4j.pax.scanner.*;
import org.osgi.framework.*;
import java.io.File;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.*;
import static org.ops4j.pax.runner.CommandLine.*;
/**
* Main runner class. Does all the work.
*
* @author Alin Dreghiciu
* @since August 26, 2007
*/
public class Run
{
/**
* Logger.
*/
private static Log LOGGER;
/**
* Handler service configuration property name.
*/
private static final String HANDLER_SERVICE = "handler.service";
/**
* Provision service configuration property name.
*/
private static final String PROVISION_SERVICE = "provision.service";
/**
* Platform extender configuration property name.
*/
private static final String PLATFORM_SERVICE = "platform.service";
/**
* Clean start configuration property name.
*/
private static final String CLEAN_START = "clean";
/**
* Working directory configuration property name.
*/
private static final String WORKING_DIRECTORY = "workingDirectory";
/**
* Handler URLs to support keepOriginalUrls option configuration property name.
*/
private static final String KEEP_ORIGINAL_HANDLER_URLS = "keep.original.handler.urls";
/**
* Creates a new runner.
*/
public Run()
{
if( LOGGER == null )
{
createLogger();
}
}
/**
* {@inheritDoc}
*/
public static void main( final String... args )
{
try
{
main( null, args );
}
catch( Throwable t )
{
showError( t );
System.exit( 1 );
}
}
/**
* Start OSGi framework based on command-line arguments, using external Java runner service.
*
* @param runner java runner service
* @param args command-line arguments
*/
public static void main( final JavaRunner runner, final String... args )
{
final CommandLine commandLine = new CommandLineImpl( args );
boolean disableLogo = Boolean.valueOf( commandLine.getOption( OPTION_NOLOGO ) );
if ( !disableLogo )
{
showLogo();
}
initializeLogger( commandLine );
String configURL = commandLine.getOption( OPTION_CONFIG );
if( configURL == null )
{
configURL = "classpath:META-INF/runner.properties";
}
final Configuration config = new ConfigurationImpl( configURL );
new Run().start(
commandLine,
config,
new OptionResolverImpl( commandLine, config ),
runner
);
}
public static Log getLogger()
{
createLogger();
return LOGGER;
}
/**
* Starts runner.
*
* @param runner java runner service
* @param args command-line arguments
*/
public static void start( final JavaRunner runner, final String... args )
{
final CommandLine commandLine = new CommandLineImpl( args );
String configURL = commandLine.getOption( OPTION_CONFIG );
if( configURL == null )
{
configURL = "classpath:META-INF/runner.properties";
}
final Configuration config = new ConfigurationImpl( configURL );
new Run().start(
commandLine,
config,
new OptionResolverImpl( commandLine, config ),
runner
);
}
/**
* Starts runner.
*
* @param commandLine comand line to use
* @param config configuration to use
* @param resolver an option resolver
*/
public void start( final CommandLine commandLine, final Configuration config, final OptionResolver resolver )
{
start( commandLine, config, resolver, null );
}
/**
* Starts runner with a java runner.
*
* @param commandLine comand line to use
* @param config configuration to use
* @param resolver an option resolver
* @param runner java runner service
*/
public void start( final CommandLine commandLine, final Configuration config, final OptionResolver resolver,
final JavaRunner runner )
{
final Context context = createContext( commandLine, config, resolver );
LOGGER.info( commandLine );
// cleanup if requested
cleanup( resolver );
// install aditional services
installServices( context );
// install aditional handlers
installHandlers( context );
// install provisioning and bundles
installBundles( installScanners( context ), new ExtensionBasedProvisionSchemaResolver(), context );
// stop the dispatcher as there are no longer events around
EventDispatcher.shutdown();
// install platform and start it up
startPlatform( installPlatform( context ), context, runner == null ? createJavaRunner( resolver ) : runner );
}
/**
* Removes the working directory if option specified.
*
* @param resolver option resolver
*/
void cleanup( final OptionResolver resolver )
{
final boolean cleanStart = Boolean.valueOf( resolver.get( CLEAN_START ) );
if( cleanStart )
{
final File workingDir = new File( resolver.getMandatory( WORKING_DIRECTORY ) );
LOGGER.debug( "Removing working directory [" + workingDir.getAbsolutePath() + "]" );
FileUtils.delete( workingDir );
}
}
/**
* Creates and initialize the context.
*
* @param commandLine comand line to use
* @param config configuration to use
* @param resolver an option resolver
*
* @return the created context
*/
Context createContext( final CommandLine commandLine, final Configuration config, final OptionResolver resolver )
{
NullArgumentException.validateNotNull( commandLine, "Command line" );
NullArgumentException.validateNotNull( config, "Configuration" );
NullArgumentException.validateNotNull( resolver, "PropertyResolver" );
final ServiceRegistry serviceRegistry = new ServiceRegistry( null );
final EventDispatcher dispatcher = EventDispatcher.start( new Logger( Logger.LOG_DEBUG ) );
serviceRegistry.addServiceListener( new ServiceListener()
{
public void serviceChanged( ServiceEvent event )
{
dispatcher.fireServiceEvent( event );
}
}
);
return new ContextImpl()
.setCommandLine( commandLine )
.setConfiguration( config )
.setOptionResolver( resolver )
.setServiceRegistry( serviceRegistry )
.setEventDispatcher( dispatcher );
}
/**
* Creates a Java runner based on "runner" option.
*
* @param resolver an option resolver
*
* @return a java runner
*/
JavaRunner createJavaRunner( final OptionResolver resolver )
{
NullArgumentException.validateNotNull( resolver, "PropertyResolver" );
LOGGER.debug( "Creating Java Runner" );
final String executor = resolver.get( OPTION_EXECUTOR );
if( executor == null || executor.trim().length() == 0 )
{
LOGGER.debug( "Using default executor" );
return null;
}
else if( "noop".equalsIgnoreCase( executor ) )
{
LOGGER.debug( "Using noop executor" );
return new NoopJavaRunner();
}
else if( "script".equalsIgnoreCase( executor ) )
{
LOGGER.debug( "Using script executor" );
return new ScriptJavaRunner();
}
else if( "zip".equalsIgnoreCase( executor ) )
{
LOGGER.debug( "Using zip executor" );
return new ZipJavaRunner();
}
else if( "inProcess".equalsIgnoreCase( executor ) )
{
LOGGER.debug( "Using in process executor" );
return new InProcessJavaRunner();
}
else if( executor.startsWith("init.d" ))
{
String[] data = executor.split(",");
String appName = (data.length > 1 && data[1].length() > 0) ? data[1] : null;
return new InitDScriptRunner(appName);
}
else if( executor.startsWith("daemon-start" ))
{
LOGGER.debug( "Using daemon-start executor" );
return new DaemonStartRunner(resolver.get(OPTION_DAEMON_TIMEOUT));
}
else if( executor.startsWith("daemon-stop" ))
{
LOGGER.debug( "Using daemon-stop executor" );
return new DaemonStopRunner();
}
else
{
try
{
final JavaRunner javaRunner = (JavaRunner) getClass().getClassLoader().loadClass( executor ).newInstance();
LOGGER.debug( "Using " + executor + " executor" );
return javaRunner;
}
catch( Exception ignore )
{
LOGGER.debug( "Connot load executor: " + executor + " reason: " + ignore.getMessage() );
}
}
throw new ConfigurationException( "Executor [" + executor + "] is not supported" );
}
/**
* Installs url handler service configured handlers.
*
* @param context the running context
*/
void installHandlers( final Context context )
{
LOGGER.debug( "Installing handlers" );
final String option = context.getOptionResolver().get( OPTION_HANDLERS );
if( option != null )
{
// first install each handler
final Configuration config = context.getConfiguration();
final String[] segments = option.split( "," );
for( String segment : segments )
{
NullArgumentException.validateNotEmpty( segment, "Handler entry" );
LOGGER.debug( "Handler [" + segment + "]" );
final String activatorName = config.getProperty( segment );
if( activatorName == null || activatorName.trim().length() == 0 )
{
throw new ConfigurationException( "Handler [" + segment + "] is not supported" );
}
createActivator( segment, activatorName, context );
}
// then install the handler service
// maintain this order as in this way the bundle context will be easier to respond to getServiceListeners
final String serviceActivatorName = config.getProperty( HANDLER_SERVICE );
if( serviceActivatorName == null || serviceActivatorName.trim().length() == 0 )
{
throw new ConfigurationException( "Handler Service must be configured [" + HANDLER_SERVICE + "]" );
}
createActivator( HANDLER_SERVICE, serviceActivatorName, context );
}
}
/**
* Installs provisioning service and configured scanners.
*
* @param context the running context
*
* @return installed provision service
*/
ProvisionService installScanners( final Context context )
{
LOGGER.debug( "Installing provisioning" );
final String option = context.getOptionResolver().getMandatory( OPTION_SCANNERS );
// first install a dummy start level service that will record the start level set by scanners
RunnerStartLevel.install( context.getServiceRegistry() );
// then install each scanner
final String[] segments = option.split( "," );
for( String segment : segments )
{
NullArgumentException.validateNotEmpty( segment, "Scanner entry" );
LOGGER.debug( "Scanner [" + segment + "]" );
final String activatorName = context.getConfiguration().getProperty( segment );
if( activatorName == null || activatorName.trim().length() == 0 )
{
throw new ConfigurationException( "Scanner [" + segment + "] is not supported" );
}
createActivator( segment, activatorName, context );
}
// then install the provisioning service
// maintain this order as in this way the bundle context will be easier to respond to getServiceListeners
final String serviceActivatorName = context.getConfiguration().getProperty( PROVISION_SERVICE );
if( serviceActivatorName == null || serviceActivatorName.trim().length() == 0 )
{
throw new ConfigurationException( "Provision Service must be configured [" + PROVISION_SERVICE + "]" );
}
final BundleContext bundleContext = createActivator( PROVISION_SERVICE, serviceActivatorName, context );
// sanity check
if( bundleContext == null )
{
throw new RuntimeException( "Could not create bundle context for provision service" );
}
final ServiceReference reference = bundleContext.getServiceReference( ProvisionService.class.getName() );
if( reference == null )
{
throw new RuntimeException( "Could not resolve a provision service" );
}
return (ProvisionService) bundleContext.getService( reference );
}
/**
* Installs additional services.
*
* @param context the running context
*/
void installServices( final Context context )
{
LOGGER.debug( "Installing additional services" );
final String option = context.getOptionResolver().get( OPTION_SERVICES );
if( option != null )
{
final Configuration config = context.getConfiguration();
final String[] segments = option.split( "," );
for( String segment : segments )
{
NullArgumentException.validateNotEmpty( segment, "Service entry" );
LOGGER.debug( "Installing service [" + segment + "]" );
final String activatorName = config.getProperty( segment );
if( activatorName == null || activatorName.trim().length() == 0 )
{
throw new ConfigurationException( "Service [" + segment + "] is not supported" );
}
createActivator( segment, activatorName, context );
}
}
}
/**
* By using provision service it installs provisioned bundles.
*
* @param provisionService installed provision service
* @param schemaResolver a provision schema resolver
* @param context the running context
*/
void installBundles( final ProvisionService provisionService,
final ProvisionSchemaResolver schemaResolver,
final Context context )
{
if( provisionService == null )
{
throw new RuntimeException( "Could not resolve a provision service" );
}
// build list of provisioning specs out of command line arguments and profiles
final List provisionSpecs = new ArrayList();
provisionSpecs.addAll( context.getCommandLine().getArguments() );
provisionSpecs.addAll( transformProfilesToProvisionSpecs( context ) );
// backup properties and replace them with audited properties
final Properties sysPropsBackup = System.getProperties();
try
{
context.setSystemProperties( new AuditedProperties( sysPropsBackup ) );
System.setProperties( context.getSystemProperties() );
final Set scannedBundles = new HashSet();
// then scan those url's
for( String provisionSpec : provisionSpecs )
{
try
{
try
{
provisionService.wrap(
filterUnique( scannedBundles, provisionService.scan( provisionSpec ) )
).install();
}
catch( UnsupportedSchemaException e )
{
final String resolvedProvisionURL = schemaResolver.resolve( provisionSpec );
if( resolvedProvisionURL != null && !resolvedProvisionURL.equals( provisionSpec ) )
{
provisionService.wrap(
filterUnique( scannedBundles, provisionService.scan( resolvedProvisionURL ) )
).install();
}
else
{
throw e;
}
}
}
catch( MalformedSpecificationException e )
{
throw new RuntimeException( e );
}
catch( ScannerException e )
{
throw new RuntimeException( e );
}
catch( BundleException e )
{
throw new RuntimeException( e );
}
}
}
finally
{
// restore the backup-ed properties
System.setProperties( sysPropsBackup );
}
}
/**
* Transforms requested profiles (--profiles option) to provisioning specs (scan-composite).
*
* @param context runner context
*
* @return list of transformed provisioning specs or an empty list if there are no profiles.
*/
private List transformProfilesToProvisionSpecs( final Context context )
{
final List provisionSpecs = new ArrayList();
final String profilesOption = context.getOptionResolver().get( OPTION_PROFILES );
if( profilesOption != null && profilesOption.trim().length() > 0 )
{
final String profilesGroup = context.getOptionResolver().get( OPTION_PROFILES_GROUPID );
final String[] profiles = profilesOption.split( ":" );
for( String profile : profiles )
{
// TODO Maybe a nice/safe parsing of profile name into group/artifact/version ?
final int startOfOptions = profile.indexOf( org.ops4j.pax.scanner.ServiceConstants.SEPARATOR_OPTION );
String options = null;
if( startOfOptions > 0 )
{
options = profile.substring( startOfOptions );
profile = profile.substring( 0, startOfOptions );
}
final String[] parts = profile.split( "/" );
provisionSpecs.add(
new StringBuilder()
.append( org.ops4j.pax.scanner.composite.ServiceConstants.SCHEMA )
.append( org.ops4j.pax.scanner.ServiceConstants.SEPARATOR_SCHEME )
.append( org.ops4j.pax.url.mvn.ServiceConstants.PROTOCOL )
.append( ":" )
.append( parts.length < 3 ? profilesGroup + "/" : "" )
.append( profile )
.append( parts.length < 2 ? "/" : "" )
.append( "/composite" )
.append( options != null ? options : "" )
.toString()
);
}
}
return provisionSpecs;
}
/**
* Filter all scanned bundles that already exists in the provided set and add the unique ones to the set.
*
* @param alreadyScanned set of already scanned bundles
* @param scannedBundles to be filtered
*
* @return unique list of scanned bundles (that were not already present in the provided set)
*/
private List filterUnique( final Set alreadyScanned,
final List scannedBundles )
{
final Set unique = new LinkedHashSet( scannedBundles );
unique.removeAll( alreadyScanned );
alreadyScanned.addAll( unique );
return new ArrayList( unique );
}
/**
* Installs platform extender and configured platform.
*
* @param context the running context
*
* @return installed platform
*/
Platform installPlatform( final Context context )
{
LOGGER.debug( "Installing platform" );
// first install platform
final String platform = context.getOptionResolver().getMandatory( OPTION_PLATFORM );
String version;
if( Boolean.parseBoolean( context.getOptionResolver().get( OPTION_PLATFORM_VERSION_SNAPSHOT ) ) )
{
version = PLATFORM_VERSION_SNAPSHOT;
}
else
{
version = context.getOptionResolver().get( OPTION_PLATFORM_VERSION );
}
if( version == null )
{
version = context.getOptionResolver().get( platform + "." + OPTION_PLATFORM_VERSION );
if( version == null )
{
throw new ConfigurationException( "Could not resolve a version for platform [" + platform + "]" );
}
}
version = version.toUpperCase();
final String activatorName = context.getConfiguration().getProperty( platform + "." + version );
if( activatorName == null || activatorName.trim().length() == 0 )
{
throw new ConfigurationException( "Platform [" + platform + " " + version + "] is not supported" );
}
createActivator( platform, activatorName, context );
// then install platform service
final String serviceActivatorName = context.getConfiguration().getProperty( PLATFORM_SERVICE );
if( serviceActivatorName == null || serviceActivatorName.trim().length() == 0 )
{
throw new ConfigurationException( "Platform Service must be configured [" + PLATFORM_SERVICE + "]" );
}
final BundleContext bundleContext = createActivator( PLATFORM_SERVICE, serviceActivatorName, context );
// sanity check
if( bundleContext == null )
{
throw new RuntimeException( "Could not create bundle context for platform service" );
}
final ServiceReference[] references;
try
{
references = bundleContext.getServiceReferences( Platform.class.getName(), "(version=" + version + ")" );
}
catch( InvalidSyntaxException ignore )
{
// this should never happen
throw new ConfigurationException( "Platform [" + platform + " " + version + "] is not supported" );
}
if( references == null || references.length == 0 )
{
throw new RuntimeException( "Could not resolve a platform" );
}
final ServiceReference reference = references[ 0 ];
return (Platform) bundleContext.getService( reference );
}
/**
* Starts the installed platform.
*
* @param context the running context
* @param platform installed platform
* @param runner Java runner service
*/
private void startPlatform( final Platform platform, final Context context, final JavaRunner runner )
{
LOGGER.debug( "Starting platform" );
if( platform == null )
{
throw new RuntimeException( "Could not resolve a platform" );
}
final List installedBundles = context.getBundles();
final List references = new ArrayList();
if( installedBundles != null )
{
for( RunnerBundle bundle : installedBundles )
{
LOGGER.info( "Provision bundle [" + bundle + "]" );
references.add(
new BundleReferenceBean(
bundle.getLocationAsURL().toExternalForm(),
bundle.getLocationAsURL(),
bundle.getStartLevel(),
bundle.shouldStart(),
bundle.shouldUpdate()
)
);
}
}
try
{
platform.start( determineSystemFiles( context ), references, context.getSystemProperties(), null, runner );
}
catch( PlatformException e )
{
throw new RuntimeException( e );
}
}
List determineSystemFiles( final Context context )
{
final List systemFiles = new ArrayList();
try
{
final String[] bcppUrls = context.getOptionResolver().getMultiple( CommandLine.OPTION_BOOT_CP_PREPEND );
if( bcppUrls.length > 0 )
{
for( String url : bcppUrls )
{
systemFiles.add( new SystemFileReferenceBean( url, new URL( url ), true ) );
}
}
final String[] bcpaUrls = context.getOptionResolver().getMultiple( CommandLine.OPTION_BOOT_CP_APPEND );
if( bcpaUrls.length > 0 )
{
for( String url : bcpaUrls )
{
systemFiles.add( new SystemFileReferenceBean( url, new URL( url ), false ) );
}
}
String keepOriginalUrls = context.getOptionResolver().get(CommandLine.OPTION_KEEP_ORIGINAL_URLS);
if ( Boolean.valueOf(keepOriginalUrls) )
{
Configuration configuration = context.getConfiguration();
String handlerUrls = configuration.getProperty(KEEP_ORIGINAL_HANDLER_URLS);
if ( handlerUrls != null )
{
String[] urlKeys = handlerUrls.split(",");
if ( urlKeys != null )
{
for ( String key : urlKeys)
{
NullArgumentException.validateNotEmpty(key, "Handler URL entry");
String handlerUrl = configuration.getProperty(key);
LOGGER.debug( "Handler URL [" + handlerUrl + "]" );
if( handlerUrl == null || handlerUrl.trim().length() == 0 )
{
throw new ConfigurationException( "Handler URL [" + key + "] is not supported" );
}
systemFiles.add( new SystemFileReferenceBean(key, new URL( handlerUrl ) ) );
}
}
}
}
}
catch( MalformedURLException e )
{
throw new RuntimeException( e );
}
return systemFiles;
}
/**
* Activator factory method.
*
* @param bundleName name of the bundle to be created
* @param activatorClazz class name of the activator
* @param context the running context
*
* @return activator related bundle context
*/
BundleContext createActivator( final String bundleName, final String activatorClazz, final Context context )
{
try
{
final BundleActivator activator = (BundleActivator) Class.forName( activatorClazz ).newInstance();
final BundleContext bundleContext = new RunnerBundleContext( context );
activator.start( bundleContext );
return bundleContext;
}
catch( Exception e )
{
throw new RuntimeException( "Could not create [" + bundleName + "]", e );
}
}
/**
* Display ops4j logo to console.
*/
private static void showLogo()
{
System.out.println( "__________ __________ " );
System.out.println( "\\______ \\_____ ___ ___ \\______ \\__ __ ____ ____ ___________" );
System.out.println( "| ___/\\__ \\ \\ \\/ / | _/ | \\/ \\ / \\_/ __ \\_ __ \\" );
System.out.println( "| | / __ \\_> < | | \\ | / | \\ | \\ ___/| | \\/" );
System.out.println( "|____| (____ /__/\\_ \\ |____|_ /____/|___| /___| /\\___ >__| " );
System.out.println( " \\/ \\/ \\/ \\/ \\/ \\/ " );
System.out.println();
final String logo = "Pax Runner " + getVersion() + "from OPS4J - http://www.ops4j.org";
System.out.println( logo );
System.out.println(
"--------------------------------------------------------------------------------------------------------"
.substring( 0, logo.length() )
);
System.out.println();
}
/**
* Discovers the Pax Runner version. If version cannot be determined returns an empty string.
*
* @return pax runner version
*/
private static String getVersion()
{
try
{
final InputStream is = Run.class.getClassLoader().getResourceAsStream( "META-INF/runner.version" );
if( is != null )
{
final Properties properties = new Properties();
properties.load( is );
final String version = properties.getProperty( "version" );
if( version != null )
{
return "(" + version + ") ";
}
return "";
}
return "";
}
catch( Exception ignore )
{
return "";
}
}
/**
* Show execution problem to console.
*
* @param t the problem
*/
private static void showError( Throwable t )
{
Info.println();
String message = t.getMessage();
String debugInfo = "";
if( LOGGER != null && !LOGGER.isDebugEnabled() )
{
debugInfo = "Use --" + OPTION_LOG + "=debug to see details.";
}
System.out.println( " ___" );
System.out.println( " / /" );
System.out.println( " / / Oops, there has been a problem!" );
System.out.println( " / / " );
System.out.println( " /__/ " + message );
System.out.println( " ___" );
System.out.println( " /__/ " + debugInfo );
System.out.println();
if( LOGGER == null )
{
// show error even when LOGGER was not initialised
System.out.println( "Exception caught during execution:" );
t.printStackTrace();
}
else
{
if( LOGGER.isDebugEnabled() )
{
LOGGER.error( "Exception caught during execution:", t );
}
}
}
/**
* Initialize the logger based on option "debug".
*
* @param commandLine command lin ein use
*/
private static void initializeLogger( final CommandLine commandLine )
{
String debug = commandLine.getOption( OPTION_LOG );
if( debug != null )
{
try
{
createLogger( LogLevel.valueOf( debug.toUpperCase() ) );
}
catch( Exception ignore )
{
createLogger( LogLevel.INFO );
LOGGER.warn( "Unknown debug option [" + debug + "], switching to INFO" );
}
}
else
{
createLogger( LogLevel.INFO );
}
}
/**
* Creates the logger to use at the specified log level. The log level is only supported by the "special" JCL
* implementation embedded into Pax Runner. In case that the JCL in the classpath in snot the embedded one it will
* fallback to standard JCL usage.
*
* @param logLevel log level to use
*/
private static void createLogger( final LogLevel logLevel )
{
try
{
LOGGER = LogFactory.getLog( Run.class, logLevel );
}
catch( NoSuchMethodError ignore )
{
// fall back to standard JCL
LOGGER = LogFactory.getLog( Run.class );
}
}
/**
* Creates a default logger at INFo level.
*/
private static void createLogger()
{
try
{
createLogger( LogLevel.INFO );
}
catch( NoClassDefFoundError ignore )
{
// fall back to standard JCL
LOGGER = LogFactory.getLog( Run.class );
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy