package org.apache.wicket.examples.sri;
import java.io.IOException;
import java.io.Serializable;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Base64;
import java.util.Base64.Encoder;
import java.util.Map;
import org.apache.commons.collections4.map.HashedMap;
import org.apache.wicket.markup.head.IHeaderResponse;
import org.apache.wicket.markup.head.IReferenceHeaderItem;
import org.apache.wicket.markup.head.ISubresourceHeaderItem;
import org.apache.wicket.markup.head.filter.SubresourceHeaderResponse;
import org.apache.wicket.request.Url;
import org.apache.wicket.request.cycle.RequestCycle;
import org.apache.wicket.request.resource.IResource;
import org.apache.wicket.request.resource.ResourceReference;
import org.apache.wicket.request.resource.caching.IStaticCacheableResource;
import org.apache.wicket.util.io.IOUtils;
import org.apache.wicket.util.resource.IResourceStream;
import org.apache.wicket.util.resource.ResourceStreamNotFoundException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@link
@author
public class DynamicSubresourceIntegrity
{
private static final Logger log = LoggerFactory.getLogger(DynamicSubresourceIntegrity.class);
private Map<Serializable, String> cache = new HashedMap<>();
@param
@return
public IHeaderResponse wrap(IHeaderResponse response)
{
return new SubresourceHeaderResponse(response)
{
@Override
protected void configure(ISubresourceHeaderItem item)
{
String integrity = getIntegrity(item);
if (integrity != null)
{
item.setIntegrity(integrity);
}
}
};
}
public String getIntegrity(ISubresourceHeaderItem item)
{
if (item instanceof IReferenceHeaderItem)
{
ResourceReference reference = ((IReferenceHeaderItem)item).getReference();
IResource resource = reference.getResource();
if (resource instanceof IStaticCacheableResource)
{
IStaticCacheableResource cacheableResource = (IStaticCacheableResource)resource;
return getIntegrity(reference, cacheableResource);
}
}
return null;
}
private String getIntegrity(ResourceReference reference, IStaticCacheableResource cacheableResource)
{
String integrity = cache.get(cacheableResource.getCacheKey());
if (integrity == null)
{
Url baseUrl = getBaseUrl(reference);
try
{
byte[] bytes = getBytes(cacheableResource, baseUrl);
integrity = "sha384-" + createHash(bytes);
cache.put(cacheableResource.getCacheKey(), integrity);
}
catch (Exception ex)
{
log.error("cannot calculate integrity", ex);
}
}
return integrity;
}
private Url getBaseUrl(ResourceReference reference)
{
RequestCycle cycle = RequestCycle.get();
Url url = Url.parse(cycle.urlFor(reference, null));
if (url.getSegments().get(0).equals("."))
{
url.removeLeadingSegments(1);
}
return url;
}
protected byte[] getBytes(IStaticCacheableResource cacheableResource, Url baseUrl)
throws IOException, ResourceStreamNotFoundException
{
byte[] bytes;
RequestCycle cycle = RequestCycle.get();
Url originalBaseUrl = cycle.getUrlRenderer().setBaseUrl(baseUrl);
try (IResourceStream stream = cacheableResource.getResourceStream())
{
bytes = IOUtils.toByteArray(stream.getInputStream());
}
finally
{
cycle.getUrlRenderer().setBaseUrl(originalBaseUrl);
}
return bytes;
}
protected String createHash(byte[] bytes) throws NoSuchAlgorithmException
{
MessageDigest digest = MessageDigest.getInstance("SHA-384");
Encoder encoder = Base64.getEncoder();
return encoder.encodeToString(digest.digest(bytes));
}
}