Removed a lot of cruft from this class that had accumulated over the years.
Deprecated methods like setConfigPath(File) and setConfigPath(String)
removed asymmetries like setConfigPath(String) and Path getConfigPath()
removed redundant configResource field
improved jarfile extraction to work for all string setters.
do not hot reload extracted config
fixed code style issues
This commit is contained in:
Greg Wilkins 2017-08-28 15:59:36 +10:00
parent b8e5a1f018
commit 6390b335fc
3 changed files with 74 additions and 70 deletions

View File

@ -18,17 +18,8 @@
package org.eclipse.jetty.security; package org.eclipse.jetty.security;
import org.eclipse.jetty.util.PathWatcher;
import org.eclipse.jetty.util.PathWatcher.PathWatchEvent;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.resource.JarResource;
import org.eclipse.jetty.util.resource.PathResource;
import org.eclipse.jetty.util.resource.Resource;
import org.eclipse.jetty.util.security.Credential;
import java.io.File; import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException; import java.io.IOException;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
@ -41,6 +32,17 @@ import java.util.Map;
import java.util.Properties; import java.util.Properties;
import java.util.Set; import java.util.Set;
import org.eclipse.jetty.toolchain.test.IO;
import org.eclipse.jetty.util.PathWatcher;
import org.eclipse.jetty.util.PathWatcher.PathWatchEvent;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.resource.JarFileResource;
import org.eclipse.jetty.util.resource.PathResource;
import org.eclipse.jetty.util.resource.Resource;
import org.eclipse.jetty.util.security.Credential;
/** /**
* PropertyUserStore * PropertyUserStore
* <p> * <p>
@ -59,13 +61,10 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
{ {
private static final Logger LOG = Log.getLogger(PropertyUserStore.class); private static final Logger LOG = Log.getLogger(PropertyUserStore.class);
private static final String JAR_FILE = "jar:file:";
protected Path _configPath; protected Path _configPath;
protected Resource _configResource;
protected PathWatcher pathWatcher; protected PathWatcher _pathWatcher;
protected boolean hotReload = false; // default is not to reload protected boolean _hotReload = false; // default is not to reload
protected boolean _firstLoad = true; // true if first load, false from that point on protected boolean _firstLoad = true; // true if first load, false from that point on
protected List<UserListener> _listeners; protected List<UserListener> _listeners;
@ -73,9 +72,7 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
/** /**
* Get the config (as a string) * Get the config (as a string)
* @return the config path as a string * @return the config path as a string
* @deprecated use {@link #getConfigPath()} instead
*/ */
@Deprecated
public String getConfig() public String getConfig()
{ {
if (_configPath != null) if (_configPath != null)
@ -89,16 +86,28 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
*/ */
public void setConfig(String config) public void setConfig(String config)
{ {
if (config == null)
{
_configPath = null;
return;
}
try try
{ {
Resource configResource = Resource.newResource(config); Resource configResource = Resource.newResource(config);
if (configResource.getFile() != null)
setConfigPath(configResource.getFile()); if (configResource instanceof JarFileResource)
_configPath = extractPackedFile((JarFileResource)configResource);
else if (configResource instanceof PathResource)
_configPath = ((PathResource)configResource).getPath();
else if (configResource.getFile() != null)
setConfigFile(configResource.getFile());
else else
throw new IllegalArgumentException(config+" is not a file"); throw new IllegalArgumentException(config);
} }
catch (Exception e) catch (Exception e)
{ {
_configPath = null;
throw new IllegalStateException(e); throw new IllegalStateException(e);
} }
@ -117,44 +126,33 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
* Set the Config Path from a String reference to a file * Set the Config Path from a String reference to a file
* @param configFile the config file can a be a file path or a reference to a file within a jar file <code>jar:file:</code> * @param configFile the config file can a be a file path or a reference to a file within a jar file <code>jar:file:</code>
*/ */
@Deprecated
public void setConfigPath(String configFile) public void setConfigPath(String configFile)
{ {
if (configFile == null) setConfig(configFile);
{
_configPath = null;
}
else if (new File( configFile ).exists())
{
_configPath = new File(configFile).toPath();
}
if ( !new File( configFile ).exists() && configFile.startsWith( JAR_FILE ))
{
// format of the url is jar:file:/foo/bar/beer.jar!/mountain_goat/pale_ale.txt
// ideally we'd like to extract this to Resource class?
try
{
_configPath = extractPackedFile( configFile );
}
catch ( IOException e )
{
throw new RuntimeException( "cannot extract file from url:" + configFile, e );
}
}
} }
private Path extractPackedFile( String configFile ) private Path extractPackedFile( JarFileResource configResource )
throws IOException throws IOException
{ {
int fileIndex = configFile.indexOf( "!" ); String uri = configResource.getURI().toASCIIString();
String entryPath = configFile.substring( fileIndex + 1, configFile.length() ); int colon = uri.lastIndexOf(":");
int bang_slash = uri.indexOf("!/");
if (colon<0 || bang_slash<0 || colon>bang_slash)
throw new IllegalArgumentException("Not resolved JarFile resource: "+uri);
String entry_path = uri.substring(colon+2).replace("!/","__").replace('/','_').replace('.','_');
Path tmpDirectory = Files.createTempDirectory( "users_store" ); Path tmpDirectory = Files.createTempDirectory( "users_store" );
Path extractedPath = Paths.get(tmpDirectory.toString(), entryPath); tmpDirectory.toFile().deleteOnExit();
// delete if exists as copyTo do not overwrite Path extractedPath = Paths.get(tmpDirectory.toString(), entry_path);
Files.deleteIfExists( extractedPath ); Files.deleteIfExists( extractedPath );
// delete on shutdown
extractedPath.toFile().deleteOnExit(); extractedPath.toFile().deleteOnExit();
JarResource.newResource( configFile ).copyTo( tmpDirectory.toFile() ); IO.copy(configResource.getInputStream(),new FileOutputStream(extractedPath.toFile()));
if (isHotReload())
{
LOG.warn("Cannot hot reload from packed configuration: {}",configResource);
setHotReload(false);
}
return extractedPath; return extractedPath;
} }
@ -162,7 +160,17 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
* Set the Config Path from a {@link File} reference * Set the Config Path from a {@link File} reference
* @param configFile the config file * @param configFile the config file
*/ */
@Deprecated
public void setConfigPath(File configFile) public void setConfigPath(File configFile)
{
setConfigFile(configFile);
}
/**
* Set the Config Path from a {@link File} reference
* @param configFile the config file
*/
public void setConfigFile(File configFile)
{ {
if(configFile == null) if(configFile == null)
{ {
@ -182,19 +190,15 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
_configPath = configPath; _configPath = configPath;
} }
/* ------------------------------------------------------------ */
/** /**
* @return the resource associated with the configured properties file, creating it if necessary * @return the resource associated with the configured properties file, creating it if necessary
* @throws IOException if unable to get the resource * @throws IOException if unable to get the resource
*/ */
public Resource getConfigResource() throws IOException public Resource getConfigResource() throws IOException
{ {
if (_configResource == null) if (_configPath==null)
{ return null;
_configResource = new PathResource(_configPath); return new PathResource(_configPath);
}
return _configResource;
} }
/** /**
@ -204,7 +208,7 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
*/ */
public boolean isHotReload() public boolean isHotReload()
{ {
return hotReload; return _hotReload;
} }
/** /**
@ -218,7 +222,7 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
{ {
throw new IllegalStateException("Cannot set hot reload while user store is running"); throw new IllegalStateException("Cannot set hot reload while user store is running");
} }
this.hotReload = enable; this._hotReload = enable;
} }
@ -247,8 +251,9 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
} }
Properties properties = new Properties(); Properties properties = new Properties();
if (getConfigResource().exists()) Resource config = getConfigResource();
properties.load(getConfigResource().getInputStream()); if (config!=null && config.exists())
properties.load(config.getInputStream());
Set<String> known = new HashSet<>(); Set<String> known = new HashSet<>();
@ -297,7 +302,6 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
} }
} }
/* /*
* set initial load to false as there should be no more initial loads * set initial load to false as there should be no more initial loads
*/ */
@ -324,11 +328,11 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
loadUsers(); loadUsers();
if ( isHotReload() && (_configPath != null) ) if ( isHotReload() && (_configPath != null) )
{ {
this.pathWatcher = new PathWatcher(); this._pathWatcher = new PathWatcher();
this.pathWatcher.watch(_configPath); this._pathWatcher.watch(_configPath);
this.pathWatcher.addListener(this); this._pathWatcher.addListener(this);
this.pathWatcher.setNotifyExistingOnStart(false); this._pathWatcher.setNotifyExistingOnStart(false);
this.pathWatcher.start(); this._pathWatcher.start();
} }
} }
@ -357,9 +361,9 @@ public class PropertyUserStore extends UserStore implements PathWatcher.Listener
protected void doStop() throws Exception protected void doStop() throws Exception
{ {
super.doStop(); super.doStop();
if (this.pathWatcher != null) if (this._pathWatcher != null)
this.pathWatcher.stop(); this._pathWatcher.stop();
this.pathWatcher = null; this._pathWatcher = null;
} }
/** /**

View File

@ -189,7 +189,7 @@ public class PropertyUserStoreTest
final String usersFile = initUsersPackedFileText(); final String usersFile = initUsersPackedFileText();
PropertyUserStore store = new PropertyUserStore(); PropertyUserStore store = new PropertyUserStore();
store.setConfigPath(usersFile); store.setConfig(usersFile);
store.registerUserListener(userCount); store.registerUserListener(userCount);

View File

@ -34,7 +34,7 @@ import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger; import org.eclipse.jetty.util.log.Logger;
/* ------------------------------------------------------------ */ /* ------------------------------------------------------------ */
class JarFileResource extends JarResource public class JarFileResource extends JarResource
{ {
private static final Logger LOG = Log.getLogger(JarFileResource.class); private static final Logger LOG = Log.getLogger(JarFileResource.class);
private JarFile _jarFile; private JarFile _jarFile;