com.googlecode.fightinglayoutbugs.ScreenshotCache.java Source code

Java tutorial

Introduction

Here is the source code for com.googlecode.fightinglayoutbugs.ScreenshotCache.java

Source

/*
 * Copyright 2009-2012 Michael Tamm
 *
 * 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.
 */

package com.googlecode.fightinglayoutbugs;

import com.googlecode.fightinglayoutbugs.helpers.ImageHelper;
import org.apache.commons.codec.binary.Base64;
import org.openqa.selenium.OutputType;
import org.openqa.selenium.TakesScreenshot;
import org.openqa.selenium.WebDriver;

import javax.annotation.Nonnull;
import java.io.*;
import java.lang.ref.SoftReference;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import static com.google.common.io.Closeables.closeQuietly;
import static com.google.common.primitives.Bytes.asList;
import static com.googlecode.fightinglayoutbugs.ScreenshotCache.Condition.UNMODIFIED;
import static com.googlecode.fightinglayoutbugs.ScreenshotCache.Condition.WITH_ALL_TEXT_TRANSPARENT;

public class ScreenshotCache {

    public enum Condition {
        UNMODIFIED(null, false), WITH_NO_IMAGES(null, true), WITH_ALL_TEXT_WHITE("#ffffff",
                false), WITH_ALL_TEXT_BLACK("#000000", false), WITH_NO_IMAGES_AND_ALL_TEXT_WHITE("#ffffff",
                        true), WITH_NO_IMAGES_AND_ALL_TEXT_BLACK("#000000",
                                true), WITH_ALL_TEXT_TRANSPARENT("transparent", false);

        final String textColor;
        final boolean hideImages;

        private Condition(String textColor, boolean hideImages) {
            this.textColor = textColor;
            this.hideImages = hideImages;
        }
    }

    private static List<Byte> PNG_SIGNATURE = asList(
            new byte[] { (byte) 0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A });

    private final WebPage _webPage;
    private final Map<Condition, SoftReference<Screenshot>> _memoryCache = new HashMap<Condition, SoftReference<Screenshot>>();
    private final Map<Condition, File> _diskCache = new HashMap<Condition, File>();
    private boolean _textColorsBackedUp;
    private boolean _imageUrlsBackedUp;
    private String _currentTextColor;
    private boolean _imagesAreCurrentlyHidden;

    public ScreenshotCache(WebPage webPage) {
        _webPage = webPage;
    }

    public @Nonnull Screenshot getScreenshot(Condition condition) {
        SoftReference<Screenshot> softReference = _memoryCache.get(condition);
        Screenshot screenshot;
        if (softReference == null) {
            // Cache miss, take screenshot ...
            screenshot = takeScreenshot(condition);
            // ... and cache it ...
            _memoryCache.put(condition, new SoftReference<Screenshot>(screenshot));
            _diskCache.put(condition, saveToTempFile(screenshot));
        } else {
            screenshot = softReference.get();
            if (screenshot == null) {
                // screenshot in _memoryCache was garbage collected, read it from _diskCache ...
                File file = _diskCache.get(condition);
                screenshot = readFromFile(file);
                // ... and put it into _memoryCache again ...
                _memoryCache.put(condition, new SoftReference<Screenshot>(screenshot));
            }
        }
        return screenshot;
    }

    private File saveToTempFile(Screenshot screenshot) {
        try {
            File tempFile = File.createTempFile("flb-cached-screenshot-", ".ser");
            tempFile.deleteOnExit();
            FileOutputStream fileOutputStream = new FileOutputStream(tempFile);
            try {
                ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);
                try {
                    objectOutputStream.writeObject(screenshot);
                } finally {
                    try {
                        objectOutputStream.close();
                    } catch (Throwable ignored) {
                    }
                }
            } finally {
                try {
                    fileOutputStream.close();
                } catch (Throwable ignored) {
                }
            }
            return tempFile;
        } catch (IOException e) {
            throw new RuntimeException("Failed to save screenshot to temporary file.", e);
        }
    }

    private Screenshot readFromFile(File file) {
        try {
            FileInputStream fileInputStream = new FileInputStream(file);
            try {
                ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);
                try {
                    Object screenshot = objectInputStream.readObject();
                    return (Screenshot) screenshot;
                } finally {
                    closeQuietly(objectInputStream);
                }
            } finally {
                closeQuietly(fileInputStream);
            }
        } catch (ClassNotFoundException e) {
            throw new RuntimeException("Failed to read cached screenshot from " + file, e);
        } catch (IOException e) {
            throw new RuntimeException("Failed to read cached screenshot from " + file, e);
        }
    }

    /**
     * Bypasses the cache and always takes a screenshot.
     */
    @Nonnull
    public Screenshot takeScreenshot(Condition condition) {
        if (condition.hideImages && _diskCache.isEmpty()) {
            // Cache UNMODIFIED and WITH_ALL_TEXT_TRANSPARENT screenshots first,
            // because restoreImages() is an asynchronous operation for InternetExplorerDriver ...
            getScreenshot(UNMODIFIED);
            getScreenshot(WITH_ALL_TEXT_TRANSPARENT);
        }
        // Handle text color ...
        if (condition.textColor != null && !condition.textColor.equals(_currentTextColor)) {
            colorAllText(condition.textColor);
            _currentTextColor = condition.textColor;
        } else if (condition.textColor == null && _currentTextColor != null) {
            restoreTextColors();
            _currentTextColor = null;
        }
        // Handle images ...
        if (condition.hideImages && !_imagesAreCurrentlyHidden) {
            hideImages();
            _imagesAreCurrentlyHidden = true;
        } else if (!condition.hideImages && _imagesAreCurrentlyHidden) {
            restoreImages();
            _imagesAreCurrentlyHidden = false;
        }
        return takeScreenshot();
    }

    void hideImages() {
        if (!_imageUrlsBackedUp) {
            _webPage.executeJavaScript("jQuery('*').each(function() {\n" + "    var $x = jQuery(this);\n"
                    + "    var b = $x.css('background-image');\n" + "    if (b && b != 'none') {\n"
                    + "        $x.data('flb_background-image_backup', b)\n"
                    + "          .css('background-image', 'url(\"\")');\n"
                    + "    }\n" + "});\n" + "jQuery('img').each(function() {\n" + "    var $img = jQuery(this);\n"
                    + "    var w = $img.width();\n" + "    var h = $img.height();\n"
                    + "    $img.data('flb_src_backup', $img.attr('src'))\n" + "        .css('width', w + 'px')\n"
                    + "        .css('height', h + 'px')\n"
                    + "        .attr('src', '')\n"
                    + "}).size();" // ... the trailing ".size()" will reduce the size of the response
            );
            _imageUrlsBackedUp = true;
        } else {
            _webPage.executeJavaScript("jQuery('*').each(function() {\n" + "    var $x = jQuery(this);\n"
                    + "    if ($x.data('flb_background-image_backup')) $x.css('background-image', 'url(\"\")');\n"
                    + "});\n" + "jQuery('img').each(function() {\n"
                    + "    jQuery(this).attr('src', '')\n"
                    + "}).size();" // ... the trailing ".size()" will reduce the size of the response
            );
        }
    }

    void restoreImages() {
        _webPage.executeJavaScript("jQuery('*').each(function() {\n" + "    var $x = jQuery(this);\n"
                + "    var b = $x.data('flb_background-image_backup');\n"
                + "    if (b) $x.css('background-image', b);\n" + "});\n" + "\n"
                + "jQuery('img').each(function() {\n" + "    var $img = jQuery(this);\n"
                + "    $img.attr('src', $img.data('flb_src_backup'));\n" + "}).size();" // ... the trailing ".size()" will reduce the size of the response
        );
    }

    void colorAllText(@Nonnull String color) {
        if (!_textColorsBackedUp) {
            _webPage.executeJavaScript(
                    "jQuery('*').each(function() { var $x = jQuery(this); $x.data('flb_color_backup', $x.css('color')); }).size();"); // ... the trailing ".size()" will reduce the size of the response
            _textColorsBackedUp = true;
        }
        _webPage.executeJavaScript("jQuery('*').css('color', '" + color + "').size();"); // ... the trailing ".size()" will reduce the size of the response
    }

    void restoreTextColors() {
        _webPage.executeJavaScript(
                "jQuery('*').each(function() { var $x = jQuery(this); $x.css('color', $x.data('flb_color_backup')); }).size();"); // ... the trailing ".size()" will reduce the size of the response
    }

    protected Screenshot takeScreenshot() {
        WebDriver driver = _webPage.getDriver();
        if (driver instanceof TakesScreenshot) {
            return takeScreenshot((TakesScreenshot) driver);
        }
        throw new UnsupportedOperationException(
                driver.getClass().getName() + " does not support taking screenshots.");
    }

    private Screenshot takeScreenshot(TakesScreenshot driver) {
        byte[] bytes = driver.getScreenshotAs(OutputType.BYTES);
        if (bytes == null) {
            throw new RuntimeException(
                    driver.getClass().getName() + ".getScreenshotAs(OutputType.BYTES) returned null.");
        }
        if (bytes.length < 8) {
            throw new RuntimeException(
                    driver.getClass().getName() + ".getScreenshotAs(OutputType.BYTES) did not return a PNG image.");
        } else {
            // Workaround for http://code.google.com/p/selenium/issues/detail?id=1686 ...
            if (!asList(bytes).subList(0, 8).equals(PNG_SIGNATURE)) {
                bytes = Base64.decodeBase64(bytes);
            }
            if (!asList(bytes).subList(0, 8).equals(PNG_SIGNATURE)) {
                throw new RuntimeException(driver.getClass().getName()
                        + ".getScreenshotAs(OutputType.BYTES) did not return a PNG image.");
            }
        }
        int[][] pixels = ImageHelper.pngToPixels(bytes);
        return new Screenshot(pixels);
    }
}