package org.jboss.security.auth.login;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.MalformedURLException;
import java.net.URL;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.ArrayList;
import javax.security.auth.AuthPermission;
import javax.security.auth.login.AppConfigurationEntry;
import javax.security.auth.login.Configuration;
import org.jboss.logging.Logger;
import org.jboss.security.auth.spi.UsersObjectModelFactory;
import org.jboss.xml.binding.JBossXBException;
import org.jboss.xml.binding.Unmarshaller;
public class XMLLoginConfigImpl extends Configuration
{
private static final String DEFAULT_APP_CONFIG_NAME = "other";
private static final AuthPermission REFRESH_PERM = new AuthPermission("refreshLoginConfiguration");
private static Logger log = Logger.getLogger(XMLLoginConfigImpl.class);
PolicyConfig appConfigs = new PolicyConfig();
protected URL loginConfigURL;
protected Configuration parentConfig;
private boolean validateDTD = true;
public void refresh()
{
SecurityManager sm = System.getSecurityManager();
if (sm != null)
sm.checkPermission(REFRESH_PERM);
if (log.isTraceEnabled())
log.trace("Begin refresh");
appConfigs.clear();
loadConfig();
if (log.isTraceEnabled())
log.trace("End refresh");
}
public AppConfigurationEntry[] getAppConfigurationEntry(String appName)
{
if (log.isTraceEnabled())
log.trace("Begin getAppConfigurationEntry("+appName+"), size="+appConfigs.size());
if (loginConfigURL == null)
{
loadConfig();
}
AppConfigurationEntry[] entry = null;
AuthenticationInfo authInfo = (AuthenticationInfo) appConfigs.get(appName);
if (authInfo == null)
{
if (log.isTraceEnabled())
log.trace("getAppConfigurationEntry("+appName+"), no entry in appConfigs, tyring parentCont: "+parentConfig);
if (parentConfig != null)
entry = parentConfig.getAppConfigurationEntry(appName);
if (entry == null)
{
if (log.isTraceEnabled())
log.trace("getAppConfigurationEntry("+appName+"), no entry in parentConfig, trying: "+DEFAULT_APP_CONFIG_NAME);
}
authInfo = (AuthenticationInfo) appConfigs.get(DEFAULT_APP_CONFIG_NAME);
}
if (authInfo != null)
{
if (log.isTraceEnabled())
log.trace("End getAppConfigurationEntry("+appName+"), authInfo=" + authInfo);
final AuthenticationInfo theAuthInfo = authInfo;
PrivilegedAction action = new PrivilegedAction()
{
public Object run()
{
return theAuthInfo.copyAppConfigurationEntry();
}
};
entry = (AppConfigurationEntry[]) AccessController.doPrivileged(action);
}
else
{
if (log.isTraceEnabled())
log.trace("End getAppConfigurationEntry("+appName+"), failed to find entry");
}
return entry;
}
public URL getConfigURL()
{
return loginConfigURL;
}
public void setConfigURL(URL loginConfigURL)
{
this.loginConfigURL = loginConfigURL;
}
public void setConfigResource(String resourceName)
throws IOException
{
ClassLoader tcl = Thread.currentThread().getContextClassLoader();
loginConfigURL = tcl.getResource(resourceName);
if (loginConfigURL == null)
throw new IOException("Failed to find resource: " + resourceName);
}
public void setParentConfig(Configuration parentConfig)
{
this.parentConfig = parentConfig;
}
public boolean getValidateDTD()
{
return this.validateDTD;
}
public void setValidateDTD(boolean flag)
{
this.validateDTD = flag;
}
public void addAppConfig(String appName, AppConfigurationEntry[] entries)
{
SecurityManager sm = System.getSecurityManager();
if (sm != null)
sm.checkPermission(REFRESH_PERM);
AuthenticationInfo authInfo = new AuthenticationInfo(appName);
authInfo.setAppConfigurationEntry(entries);
if (log.isTraceEnabled())
log.trace("addAppConfig("+appName+"), authInfo=" + authInfo);
appConfigs.add(authInfo);
}
public void removeAppConfig(String appName)
{
SecurityManager sm = System.getSecurityManager();
if (sm != null)
sm.checkPermission(REFRESH_PERM);
if (log.isTraceEnabled())
log.trace("removeAppConfig, appName="+appName);
appConfigs.remove(appName);
}
public void clear()
{
}
public void loadConfig()
{
String loginConfig = System.getProperty("java.security.auth.login.config");
if (loginConfig == null)
loginConfig = "login-config.xml";
if (loginConfigURL == null)
{
try
{
loginConfigURL = new URL(loginConfig);
}
catch (MalformedURLException e)
{
try
{
setConfigResource(loginConfig);
}
catch (IOException ignore)
{
File configFile = new File(loginConfig);
try
{
setConfigURL(configFile.toURL());
}
catch (MalformedURLException ignore2)
{
}
}
}
}
if (loginConfigURL == null)
{
log.warn("Failed to find config: " + loginConfig);
return;
}
if (log.isTraceEnabled())
log.trace("Begin loadConfig, loginConfigURL="+loginConfigURL);
try
{
loadConfig(loginConfigURL);
if (log.isTraceEnabled())
log.trace("End loadConfig, loginConfigURL="+loginConfigURL);
}
catch (Exception e)
{
log.warn("End loadConfig, failed to load config: " + loginConfigURL, e);
}
}
protected String[] loadConfig(URL config) throws Exception
{
SecurityManager sm = System.getSecurityManager();
if (sm != null)
sm.checkPermission(REFRESH_PERM);
ArrayList configNames = new ArrayList();
log.debug("Try loading config as XML, url=" + config);
try
{
loadXMLConfig(config, configNames);
}
catch(Throwable e)
{
log.debug("Failed to load config as XML", e);
log.debug("Try loading config as Sun format, url=" + config);
loadSunConfig(config, configNames);
}
String[] names = new String[configNames.size()];
configNames.toArray(names);
return names;
}
private void loadSunConfig(URL sunConfig, ArrayList configNames)
throws Exception
{
InputStream is = sunConfig.openStream();
if (is == null)
throw new IOException("InputStream is null for: " + sunConfig);
InputStreamReader configFile = new InputStreamReader(is);
boolean trace = log.isTraceEnabled();
SunConfigParser.doParse(configFile, this, trace);
}
private void loadXMLConfig(URL loginConfigURL, ArrayList configNames)
throws IOException, JBossXBException
{
LoginConfigObjectModelFactory lcomf = new LoginConfigObjectModelFactory();
UsersObjectModelFactory uomf = new UsersObjectModelFactory();
InputStreamReader xmlReader = loadURL(loginConfigURL);
Unmarshaller unmarshaller = new Unmarshaller();
unmarshaller.mapFactoryToNamespace(uomf, "http://www.jboss.org/j2ee/schemas/XMLLoginModule");
PolicyConfig config = (PolicyConfig) unmarshaller.unmarshal(xmlReader, lcomf, null);
configNames.addAll(config.getConfigNames());
appConfigs.copy(config);
}
private InputStreamReader loadURL(URL configURL)
throws IOException
{
InputStream is = configURL.openStream();
if (is == null)
throw new IOException("Failed to obtain InputStream from url: " + configURL);
InputStreamReader xmlReader = new InputStreamReader(is);
return xmlReader;
}
}