AssetRendition.java
/*
* #%L
* wcm.io
* %%
* Copyright (C) 2019 wcm.io
* %%
* 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.
* #L%
*/
package io.wcm.handler.mediasource.dam;
import static com.day.cq.commons.jcr.JcrConstants.JCR_CONTENT;
import static com.day.cq.dam.api.DamConstants.EXIF_PIXELXDIMENSION;
import static com.day.cq.dam.api.DamConstants.EXIF_PIXELYDIMENSION;
import static com.day.cq.dam.api.DamConstants.METADATA_FOLDER;
import static com.day.cq.dam.api.DamConstants.ORIGINAL_FILE;
import static com.day.cq.dam.api.DamConstants.TIFF_IMAGELENGTH;
import static com.day.cq.dam.api.DamConstants.TIFF_IMAGEWIDTH;
import static io.wcm.handler.mediasource.dam.impl.metadata.RenditionMetadataNameConstants.NN_RENDITIONS_METADATA;
import static io.wcm.handler.mediasource.dam.impl.metadata.RenditionMetadataNameConstants.PN_IMAGE_HEIGHT;
import static io.wcm.handler.mediasource.dam.impl.metadata.RenditionMetadataNameConstants.PN_IMAGE_WIDTH;
import java.io.IOException;
import java.io.InputStream;
import org.apache.commons.io.FilenameUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.math.NumberUtils;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ValueMap;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.osgi.annotation.versioning.ProviderType;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.day.cq.dam.api.Asset;
import com.day.cq.dam.api.Rendition;
import com.day.image.Layer;
import io.wcm.handler.media.Dimension;
import io.wcm.handler.media.MediaFileType;
import io.wcm.sling.commons.adapter.AdaptTo;
/**
* Helper methods for getting metadata for DAM renditions.
*/
@ProviderType
public final class AssetRendition {
private static final Logger log = LoggerFactory.getLogger(AssetRendition.class);
private AssetRendition() {
// static methods only
}
/**
* Get dimension (width, height) of given DAM rendition.
*
* <p>
* It reads the dimension information from the
* asset metadata for the original rendition, or from the rendition metadata generated by the
* "DamRenditionMetadataService". If both is not available it gets the dimension from the renditions
* binary file, but this is inefficient and should not happen under sound conditions.
* </p>
* @param rendition Rendition
* @return Dimension or null if dimension could not be detected, not even in fallback mode
*/
public static @Nullable Dimension getDimension(@NotNull Rendition rendition) {
return getDimension(rendition, false);
}
/**
* Get dimension (width, height) of given DAM rendition.
*
* <p>
* It reads the dimension information from the
* asset metadata for the original rendition, or from the rendition metadata generated by the
* "DamRenditionMetadataService". If both is not available it gets the dimension from the renditions
* binary file, but this is inefficient and should not happen under sound conditions.
* </p>
* @param rendition Rendition
* @param suppressLogWarningNoRenditionsMetadata If set to true, no log warnings is generated when
* renditions metadata containing the width/height of the rendition does not exist (yet).
* @return Dimension or null if dimension could not be detected, not even in fallback mode
*/
public static @Nullable Dimension getDimension(@NotNull Rendition rendition,
boolean suppressLogWarningNoRenditionsMetadata) {
boolean isOriginal = isOriginal(rendition);
String fileExtension = FilenameUtils.getExtension(getFilename(rendition));
// get image width/height
Dimension dimension = null;
if (isOriginal) {
// get width/height from metadata for original renditions
dimension = getDimensionFromOriginal(rendition);
}
// dimensions for non-original renditions only supported for image binaries
if (MediaFileType.isImage(fileExtension)) {
if (dimension == null) {
// check if rendition metadata is present in <rendition>/jcr:content/metadata provided by AEMaaCS asset compute
dimension = getDimensionFromAemRenditionMetadata(rendition);
}
if (dimension == null) {
// otherwise get from rendition metadata written by {@link DamRenditionMetadataService}
dimension = getDimensionFromMediaHandlerRenditionMetadata(rendition);
}
// fallback: if width/height could not be read from either asset or rendition metadata load the image
// into memory and get width/height from there - but log an warning because this is inefficient
if (dimension == null) {
dimension = getDimensionFromImageBinary(rendition, suppressLogWarningNoRenditionsMetadata);
}
}
return dimension;
}
/**
* Read dimension for original rendition from asset metadata.
* @param rendition Rendition
* @return Dimension or null
*/
private static @Nullable Dimension getDimensionFromOriginal(@NotNull Rendition rendition) {
Asset asset = rendition.getAsset();
// asset may have stored dimension in different property names
long width = getAssetMetadataValueAsLong(asset, TIFF_IMAGEWIDTH, EXIF_PIXELXDIMENSION);
long height = getAssetMetadataValueAsLong(asset, TIFF_IMAGELENGTH, EXIF_PIXELYDIMENSION);
return toValidDimension(width, height);
}
private static long getAssetMetadataValueAsLong(Asset asset, String... propertyNames) {
for (String propertyName : propertyNames) {
long value = NumberUtils.toLong(StringUtils.defaultString(asset.getMetadataValueFromJcr(propertyName), "0"));
if (value > 0L) {
return value;
}
}
return 0L;
}
/**
* Read dimension for non-original rendition from renditions metadata generated by "DamRenditionMetadataService".
* @param rendition Rendition
* @return Dimension or null
*/
@SuppressWarnings("java:S1075") // not a file path
private static @Nullable Dimension getDimensionFromMediaHandlerRenditionMetadata(@NotNull Rendition rendition) {
Asset asset = rendition.getAsset();
String metadataPath = JCR_CONTENT + "/" + NN_RENDITIONS_METADATA + "/" + rendition.getName();
Resource metadataResource = AdaptTo.notNull(asset, Resource.class).getChild(metadataPath);
if (metadataResource != null) {
ValueMap props = metadataResource.getValueMap();
long width = props.get(PN_IMAGE_WIDTH, 0L);
long height = props.get(PN_IMAGE_HEIGHT, 0L);
return toValidDimension(width, height);
}
return null;
}
/**
* Asset Compute from AEMaaCS writes rendition metadata including width/height to jcr:content/metadata of the
* rendition resource - try to read it from there (it may be missing for not fully processed assets, or in local
* AEMaaCS SDK or AEM 6.5 instances).
* @param rendition Rendition
* @return Dimension or null
*/
private static @Nullable Dimension getDimensionFromAemRenditionMetadata(@NotNull Rendition rendition) {
Resource metadataResource = rendition.getChild(JCR_CONTENT + "/" + METADATA_FOLDER);
if (metadataResource != null) {
ValueMap props = metadataResource.getValueMap();
long width = props.get(TIFF_IMAGEWIDTH, 0L);
long height = props.get(TIFF_IMAGELENGTH, 0L);
return toValidDimension(width, height);
}
return null;
}
/**
* Fallback: Read dimension by loading image binary into memory.
* @param rendition Rendition
* @param suppressLogWarningNoRenditionsMetadata If set to true, no log warnings is generated when
* renditions metadata containing the width/height of the rendition does not exist (yet).
* @return Dimension or null
*/
@SuppressWarnings("PMD.GuardLogStatement")
private static @Nullable Dimension getDimensionFromImageBinary(@NotNull Rendition rendition,
boolean suppressLogWarningNoRenditionsMetadata) {
try (InputStream is = rendition.getStream()) {
if (is != null) {
Layer layer = new Layer(is);
long width = layer.getWidth();
long height = layer.getHeight();
Dimension dimension = toValidDimension(width, height);
if (!suppressLogWarningNoRenditionsMetadata) {
log.warn("Unable to detect rendition metadata for {}, "
+ "fallback to inefficient detection by loading image into in memory (detected dimension={}). "
+ "Please check if the service user for the bundle 'io.wcm.handler.media' is configured properly.",
rendition.getPath(), dimension);
}
return dimension;
}
else {
log.warn("Unable to get binary stream for rendition {}", rendition.getPath());
}
}
catch (IOException ex) {
log.warn("Unable to read binary stream to layer for rendition {}", rendition.getPath(), ex);
}
return null;
}
/**
* Convert width/height to dimension.
* @param width Width
* @param height Height
* @return Dimension or null if width or height are not valid
*/
private static @Nullable Dimension toValidDimension(long width, long height) {
if (width > 0L && height > 0L) {
return new Dimension(width, height);
}
return null;
}
/**
* Checks if the given rendition is the original file of the asset
* @param rendition DAM rendition
* @return true if rendition is the original
*/
public static boolean isOriginal(@NotNull Rendition rendition) {
return StringUtils.equals(rendition.getName(), ORIGINAL_FILE);
}
/**
* Checks if the given rendition is a thumbnail rendition generated automatically by AEM
* (with <code>cq5dam.thumbnail.</code> prefix).
* @param rendition DAM rendition
* @return true if rendition is a thumbnail rendition
*/
public static boolean isThumbnailRendition(@NotNull Rendition rendition) {
return AemRenditionType.THUMBNAIL_RENDITION.matches(rendition);
}
/**
* Checks if the given rendition is a web rendition generated automatically by AEM for the image editor/cropping
* (with <code>cq5dam.web.</code> prefix).
* @param rendition DAM rendition
* @return true if rendition is a web rendition
*/
public static boolean isWebRendition(@NotNull Rendition rendition) {
return AemRenditionType.WEB_RENDITION.matches(rendition);
}
/**
* Get file name of given rendition. If it is the original rendition get asset name as file name.
* @param rendition Rendition
* @return File extension or null if it could not be detected
*/
public static String getFilename(@NotNull Rendition rendition) {
boolean isOriginal = isOriginal(rendition);
if (isOriginal) {
return rendition.getAsset().getName();
}
else {
return rendition.getName();
}
}
}