org.opensaml.saml2.metadata.provider.AbstractReloadingMetadataProvider Maven / Gradle / Ivy
/*
* Licensed to the University Corporation for Advanced Internet Development,
* Inc. (UCAID) under one or more contributor license agreements. See the
* NOTICE file distributed with this work for additional information regarding
* copyright ownership. The UCAID 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.opensaml.saml2.metadata.provider;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Timer;
import java.util.TimerTask;
import org.joda.time.DateTime;
import org.joda.time.DateTimeZone;
import org.joda.time.chrono.ISOChronology;
import org.opensaml.saml2.common.SAML2Helper;
import org.opensaml.xml.XMLObject;
import org.opensaml.xml.io.UnmarshallingException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.w3c.dom.Document;
/**
* Base class for metadata providers that cache and periodically refresh their metadata.
*
* This metadata provider periodically checks to see if the read metadata file has changed. The delay between each
* refresh interval is calculated as follows. If no validUntil or cacheDuration is present then the
* {@link #getMaxRefreshDelay()} value is used. Otherwise, the earliest refresh interval of the metadata file is checked
* by looking for the earliest of all the validUntil attributes and cacheDuration attributes. If that refresh interval
* is larger than the max refresh delay then {@link #getMaxRefreshDelay()} is used. If that number is smaller than the
* min refresh delay then {@link #getMinRefreshDelay()} is used. Otherwise the calculated refresh delay multiplied by
* {@link #getRefreshDelayFactor()} is used. By using this factor, the provider will attempt to be refresh before the
* cache actually expires, allowing a some room for error and recovery. Assuming the factor is not exceedingly close to
* 1.0 and a min refresh delay that is not overly large, this refresh will likely occur a few times before the cache
* expires.
*/
public abstract class AbstractReloadingMetadataProvider extends AbstractObservableMetadataProvider {
/** Class logger. */
private final Logger log = LoggerFactory.getLogger(AbstractReloadingMetadataProvider.class);
/** Timer used to schedule background metadata update tasks. */
private Timer taskTimer;
/** Whether we created our own task timer during object construction. */
private boolean createdOwnTaskTimer;
/** Current task to refresh metadata. */
private RefreshMetadataTask refresMetadataTask;
/** Factor used to compute when the next refresh interval will occur. Default value: 0.75 */
private float refreshDelayFactor = 0.75f;
/**
* Refresh interval used when metadata does not contain any validUntil or cacheDuration information. Default value:
* 14400000ms
*/
private long maxRefreshDelay = 14400000;
/** Floor, in milliseconds, for the refresh interval. Default value: 300000ms */
private int minRefreshDelay = 300000;
/** Time when the currently cached metadata file expires. */
private DateTime expirationTime;
/** Last time the metadata was updated. */
private DateTime lastUpdate;
/** Last time a refresh cycle occurred. */
private DateTime lastRefresh;
/** Next time a refresh cycle will occur. */
private DateTime nextRefresh;
/** Last successfully read in metadata. */
private XMLObject cachedMetadata;
/** Constructor. */
protected AbstractReloadingMetadataProvider() {
taskTimer = new Timer(true);
createdOwnTaskTimer = true;
}
/**
* Constructor.
*
* @param backgroundTaskTimer time used to schedule background refresh tasks
*/
protected AbstractReloadingMetadataProvider(Timer backgroundTaskTimer) {
if (backgroundTaskTimer == null) {
throw new IllegalArgumentException("Task timer may not be null");
}
taskTimer = backgroundTaskTimer;
}
/**
* Gets the time when the currently cached metadata expires.
*
* @return time when the currently cached metadata expires, or null if no metadata is cached
*/
public DateTime getExpirationTime() {
return expirationTime;
}
/**
* Gets the time that the currently available metadata was last updated. Note, this may be different than the time
* retrieved by {@link #getLastRefresh()} if the metadata was known not to have changed during the last refresh
* cycle.
*
* @return time when the currently metadata was last update, null if metadata has never successfully been read in
*/
public DateTime getLastUpdate() {
return lastUpdate;
}
/**
* Gets the time the last refresh cycle occurred.
*
* @return time the last refresh cycle occurred
*/
public DateTime getLastRefresh() {
return lastRefresh;
}
/**
* Gets the time when the next refresh cycle will occur.
*
* @return time when the next refresh cycle will occur
*/
public DateTime getNextRefresh() {
return nextRefresh;
}
/**
* Gets the maximum amount of time, in milliseconds, between refresh intervals.
*
* @return maximum amount of time between refresh intervals
*/
public long getMaxRefreshDelay() {
return maxRefreshDelay;
}
/**
* Sets the maximum amount of time, in milliseconds, between refresh intervals.
*
* @param delay maximum amount of time, in milliseconds, between refresh intervals
*/
public void setMaxRefreshDelay(long delay) {
if (delay < 0) {
throw new IllegalArgumentException("Maximum refresh delay must be greater than 0");
}
maxRefreshDelay = delay;
}
/**
* Gets the delay factor used to compute the next refresh time.
*
* @return delay factor used to compute the next refresh time
*/
public float getRefreshDelayFactor() {
return refreshDelayFactor;
}
/**
* Sets the delay factor used to compute the next refresh time. The delay must be between 0.0 and 1.0, exclusive.
*
* @param factor delay factor used to compute the next refresh time
*/
public void setRefreshDelayFactor(float factor) {
if (factor <= 0 || factor >= 1) {
throw new IllegalArgumentException("Refresh delay factor must be a number between 0.0 and 1.0, exclusive");
}
refreshDelayFactor = factor;
}
/**
* Gets the minimum amount of time, in milliseconds, between refreshes.
*
* @return minimum amount of time, in milliseconds, between refreshes
*/
public int getMinRefreshDelay() {
return minRefreshDelay;
}
/**
* Sets the minimum amount of time, in milliseconds, between refreshes.
*
* @param delay minimum amount of time, in milliseconds, between refreshes
*/
public void setMinRefreshDelay(int delay) {
if (delay < 0) {
throw new IllegalArgumentException("Minimum refresh delay must be greater than 0");
}
minRefreshDelay = delay;
}
/** {@inheritDoc} */
public synchronized void destroy() {
refresMetadataTask.cancel();
if (createdOwnTaskTimer) {
taskTimer.cancel();
}
expirationTime = null;
lastRefresh = null;
lastUpdate = null;
nextRefresh = null;
cachedMetadata = null;
super.destroy();
}
/** {@inheritDoc} */
protected XMLObject doGetMetadata() throws MetadataProviderException {
return cachedMetadata;
}
/** {@inheritDoc} */
protected void doInitialization() throws MetadataProviderException {
refresh();
if (minRefreshDelay > maxRefreshDelay) {
throw new MetadataProviderException("Minimum refresh delay " + minRefreshDelay
+ " is greater than maximum refresh delay " + maxRefreshDelay);
}
}
/**
* Refreshes the metadata from its source.
*
* @throws MetadataProviderException thrown is there is a problem retrieving and processing the metadata
*/
public synchronized void refresh() throws MetadataProviderException {
DateTime now = new DateTime(ISOChronology.getInstanceUTC());
String mdId = getMetadataIdentifier();
log.debug("Beginning refresh of metadata from '{}'", mdId);
try {
byte[] mdBytes = fetchMetadata();
if (mdBytes == null) {
log.debug("Metadata from '{}' has not changed since last refresh", mdId);
processCachedMetadata(mdId, now);
} else {
log.debug("Processing new metadata from '{}'", mdId);
processNewMetadata(mdId, now, mdBytes);
}
} catch (Throwable t) {
log.debug("Error occurred while attempting to refresh metadata from '" + mdId + "'", t);
nextRefresh = new DateTime(ISOChronology.getInstanceUTC()).plus(minRefreshDelay);
if (t instanceof Exception) {
throw new MetadataProviderException((Exception) t);
} else {
throw new MetadataProviderException(String.format("Saw an error of type '%s' with message '%s'",
t.getClass().getName(), t.getMessage()));
}
} finally {
refresMetadataTask = new RefreshMetadataTask();
long nextRefreshDelay = nextRefresh.getMillis() - System.currentTimeMillis();
taskTimer.schedule(refresMetadataTask, nextRefreshDelay);
log.info("Next refresh cycle for metadata provider '{}' will occur on '{}' ('{}' local time)",
new Object[] {mdId, nextRefresh, nextRefresh.toDateTime(DateTimeZone.getDefault()),});
lastRefresh = now;
}
}
/**
* Gets an identifier which may be used to distinguish this metadata in logging statements.
*
* @return identifier which may be used to distinguish this metadata in logging statements
*/
protected abstract String getMetadataIdentifier();
/**
* Fetches metadata from a source.
*
* @return the fetched metadata, or null if the metadata is known not to have changed since the last retrieval
*
* @throws MetadataProviderException thrown if there is a problem fetching the metadata
*/
protected abstract byte[] fetchMetadata() throws MetadataProviderException;
/**
* Unmarshalls the given metadata bytes.
*
* @param metadataBytes raw metadata bytes
*
* @return the metadata
*
* @throws MetadataProviderException thrown if the metadata can not be unmarshalled
*/
protected XMLObject unmarshallMetadata(byte[] metadataBytes) throws MetadataProviderException {
try {
return unmarshallMetadata(new ByteArrayInputStream(metadataBytes));
} catch (UnmarshallingException e) {
String errorMsg = "Unable to unmarshall metadata";
log.error(errorMsg, e);
throw new MetadataProviderException(errorMsg, e);
}
}
/**
* Processes a cached metadata document in order to determine, and schedule, the next time it should be refreshed.
*
* @param metadataIdentifier identifier of the metadata source
* @param refreshStart when the current refresh cycle started
*
* @throws MetadataProviderException throw is there is a problem process the cached metadata
*/
protected void processCachedMetadata(String metadataIdentifier, DateTime refreshStart)
throws MetadataProviderException {
log.debug("Computing new expiration time for cached metadata from '{}", metadataIdentifier);
DateTime metadataExpirationTime =
SAML2Helper
.getEarliestExpiration(cachedMetadata, refreshStart.plus(getMaxRefreshDelay()), refreshStart);
expirationTime = metadataExpirationTime;
long nextRefreshDelay = computeNextRefreshDelay(expirationTime);
nextRefresh = new DateTime(ISOChronology.getInstanceUTC()).plus(nextRefreshDelay);
}
/**
* Process a new metadata document. Processing include unmarshalling and filtering metadata, determining the next
* time is should be refreshed and scheduling the next refresh cycle.
*
* @param metadataIdentifier identifier of the metadata source
* @param refreshStart when the current refresh cycle started
* @param metadataBytes raw bytes of the new metadata document
*
* @throws MetadataProviderException thrown if there is a problem unmarshalling or filtering the new metadata
*/
protected void processNewMetadata(String metadataIdentifier, DateTime refreshStart, byte[] metadataBytes)
throws MetadataProviderException {
log.debug("Unmarshalling metadata from '{}'", metadataIdentifier);
XMLObject metadata = unmarshallMetadata(metadataBytes);
if (!isValid(metadata)) {
processPreExpiredMetadata(metadataIdentifier, refreshStart, metadataBytes, metadata);
} else {
processNonExpiredMetadata(metadataIdentifier, refreshStart, metadataBytes, metadata);
}
}
/**
* Processes metadata that has been determined to be invalid (usually because it's already expired) at the time it
* was fetched. A metadata document is considered be invalid if its root element returns false when passed to the
* {@link #isValid(XMLObject)} method.
*
* @param metadataIdentifier identifier of the metadata source
* @param refreshStart when the current refresh cycle started
* @param metadataBytes raw bytes of the new metadata document
* @param metadata new metadata document unmarshalled
*/
protected void processPreExpiredMetadata(String metadataIdentifier, DateTime refreshStart, byte[] metadataBytes,
XMLObject metadata) {
log.warn("Entire metadata document from '{}' was expired at time of loading, existing metadata retained",
metadataIdentifier);
lastUpdate = refreshStart;
nextRefresh = new DateTime(ISOChronology.getInstanceUTC()).plus(getMinRefreshDelay());
}
/**
* Processes metadata that has been determined to be valid at the time it was fetched. A metadata document is
* considered be valid if its root element returns true when passed to the {@link #isValid(XMLObject)} method.
*
* @param metadataIdentifier identifier of the metadata source
* @param refreshStart when the current refresh cycle started
* @param metadataBytes raw bytes of the new metadata document
* @param metadata new metadata document unmarshalled
*
* @throws MetadataProviderException thrown if there s a problem processing the metadata
*/
protected void processNonExpiredMetadata(String metadataIdentifier, DateTime refreshStart, byte[] metadataBytes,
XMLObject metadata) throws MetadataProviderException {
Document metadataDom = metadata.getDOM().getOwnerDocument();
log.debug("Filtering metadata from '{}'", metadataIdentifier);
try {
filterMetadata(metadata);
} catch (FilterException e) {
String errMsg = "Error filtering metadata from " + metadataIdentifier;
log.error(errMsg, e);
throw new MetadataProviderException(errMsg, e);
}
log.debug("Releasing cached DOM for metadata from '{}'", metadataIdentifier);
releaseMetadataDOM(metadata);
log.debug("Post-processing metadata from '{}'", metadataIdentifier);
postProcessMetadata(metadataBytes, metadataDom, metadata);
log.debug("Computing expiration time for metadata from '{}'", metadataIdentifier);
DateTime metadataExpirationTime =
SAML2Helper.getEarliestExpiration(metadata, refreshStart.plus(getMaxRefreshDelay()), refreshStart);
log.debug("Expiration of metadata from '{}' will occur at {}", metadataIdentifier,
metadataExpirationTime.toString());
cachedMetadata = metadata;
lastUpdate = refreshStart;
long nextRefreshDelay;
if (metadataExpirationTime.isBeforeNow()) {
expirationTime = new DateTime(ISOChronology.getInstanceUTC()).plus(getMinRefreshDelay());
nextRefreshDelay = getMaxRefreshDelay();
} else {
expirationTime = metadataExpirationTime;
nextRefreshDelay = computeNextRefreshDelay(expirationTime);
}
nextRefresh = new DateTime(ISOChronology.getInstanceUTC()).plus(nextRefreshDelay);
emitChangeEvent();
log.info("New metadata succesfully loaded for '{}'", getMetadataIdentifier());
}
/**
* Post-processing hook called after new metadata has been unmarshalled, filtered, and the DOM released (from the
* {@link XMLObject}) but before the metadata is saved off. Any exception thrown by this hook will cause the
* retrieved metadata to be discarded.
*
* The default implementation of this method is a no-op
*
* @param metadataBytes raw metadata bytes retrieved via {@link #fetchMetadata}
* @param metadataDom metadata after it has been parsed in to a DOM document
* @param metadata metadata after it has been run through all registered filters and its DOM released
*
* @throws MetadataProviderException thrown if there is a problem with the provided data
*/
protected void postProcessMetadata(byte[] metadataBytes, Document metadataDom, XMLObject metadata)
throws MetadataProviderException {
}
/**
* Computes the delay until the next refresh time based on the current metadata's expiration time and the refresh
* interval floor.
*
* @param expectedExpiration the time when the metadata is expected to expire and need refreshing
*
* @return delay, in milliseconds, until the next refresh time
*/
protected long computeNextRefreshDelay(DateTime expectedExpiration) {
long now = new DateTime(ISOChronology.getInstanceUTC()).getMillis();
long expireInstant = 0;
if (expectedExpiration != null) {
expireInstant = expectedExpiration.toDateTime(ISOChronology.getInstanceUTC()).getMillis();
}
long refreshDelay = (long) ((expireInstant - now) * getRefreshDelayFactor());
// if the expiration time was null or the calculated refresh delay was less than the floor
// use the floor
if (refreshDelay < getMinRefreshDelay()) {
refreshDelay = getMinRefreshDelay();
}
return refreshDelay;
}
/**
* Converts an InputStream into a byte array.
*
* @param ins input stream to convert
*
* @return resultant byte array
*
* @throws MetadataProviderException thrown if there is a problem reading the resultant byte array
*/
protected byte[] inputstreamToByteArray(InputStream ins) throws MetadataProviderException {
try {
// 1 MB read buffer
byte[] buffer = new byte[1024 * 1024];
ByteArrayOutputStream output = new ByteArrayOutputStream();
int n = 0;
while (-1 != (n = ins.read(buffer))) {
output.write(buffer, 0, n);
}
ins.close();
return output.toByteArray();
} catch (IOException e) {
throw new MetadataProviderException(e);
}
}
/** Background task that refreshes metadata. */
private class RefreshMetadataTask extends TimerTask {
/** {@inheritDoc} */
public void run() {
try {
if (!isInitialized()) {
// just in case the metadata provider was destroyed before this task runs
return;
}
refresh();
} catch (MetadataProviderException e) {
// nothing to do, error message already logged by refreshMetadata()
return;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy