All Downloads are FREE. Search and download functionalities are using the official Maven repository.

hudson.plugins.android_emulator.AndroidEmulator Maven / Gradle / Ivy

There is a newer version: 1.6
Show newest version
package hudson.plugins.android_emulator;


import hudson.EnvVars;
import hudson.Extension;
import hudson.FilePath;
import hudson.Launcher;
import hudson.Proc;
import hudson.Util;
import hudson.Launcher.ProcStarter;
import hudson.model.AbstractBuild;
import hudson.model.AbstractProject;
import hudson.model.BuildListener;
import hudson.model.Computer;
import hudson.model.Hudson;
import hudson.model.Result;
import hudson.model.TaskListener;
import hudson.remoting.Callable;
import hudson.tasks.BuildWrapper;
import hudson.tasks.BuildWrapperDescriptor;
import hudson.util.ArgumentListBuilder;
import hudson.util.FormValidation;

import java.io.File;
import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintStream;
import java.io.Serializable;
import java.net.Socket;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import net.sf.json.JSONObject;

import org.apache.commons.io.output.ByteArrayOutputStream;
import org.jvnet.hudson.plugins.port_allocator.PortAllocationManager;
import org.kohsuke.stapler.DataBoundConstructor;
import org.kohsuke.stapler.QueryParameter;
import org.kohsuke.stapler.StaplerRequest;

public class AndroidEmulator extends BuildWrapper implements Serializable {

    private static final long serialVersionUID = 1L;

    /** Duration by which the emulator should start being available via adb. */
    private static final int ADB_CONNECT_TIMEOUT_MS = 60 * 1000;

    /** Duration by which emulator booting should normally complete. */
    private static final int BOOT_COMPLETE_TIMEOUT_MS = 120 * 1000;

    private DescriptorImpl descriptor;

    private final String avdName;
    private final String osVersion;
    private final String screenDensity;
    private final String screenResolution;
    private final String deviceLocale;
    private final String sdCardSize;
    private final boolean wipeData;
    private final boolean showWindow;
    private final int startupDelay;

    @DataBoundConstructor
    public AndroidEmulator(String avdName, String osVersion, String screenDensity,
            String screenResolution, String deviceLocale, String sdCardSize, boolean wipeData,
            boolean showWindow, int startupDelay) {
        this.avdName = avdName;
        this.osVersion = osVersion;
        this.screenDensity = screenDensity;
        this.screenResolution = screenResolution;
        this.deviceLocale = deviceLocale;
        this.sdCardSize = sdCardSize;
        this.wipeData = wipeData;
        this.showWindow = showWindow;
        this.startupDelay = Math.abs(startupDelay);
    }

    public boolean getUseNamedEmulator() {
        return avdName != null;
    }

    public String getOsVersion() {
        return osVersion;
    }

    public String getAvdName() {
        return avdName;
    }

    public String getScreenDensity() {
        return screenDensity;
    }

    public String getScreenResolution() {
        return screenResolution;
    }

    public String getDeviceLocale() {
        return deviceLocale;
    }

    public String getSdCardSize() {
        return sdCardSize;
    }

    public boolean shouldWipeData() {
        return wipeData;
    }

    public boolean shouldShowWindow() {
        return showWindow;
    }

    public int getStartupDelay() {
        return startupDelay;
    }

    @Override
    @SuppressWarnings("unchecked")
    public Environment setUp(AbstractBuild build, final Launcher launcher, BuildListener listener)
            throws IOException, InterruptedException {
        final PrintStream logger = listener.getLogger();
        if (descriptor == null) {
            descriptor = Hudson.getInstance().getDescriptorByType(DescriptorImpl.class);
        }

        // Substitute environment and build variables into config
        final EnvVars localVars = Computer.currentComputer().getEnvironment();
        final EnvVars envVars = new EnvVars(localVars);
        envVars.putAll(build.getEnvironment(listener));
        final Map buildVars = build.getBuildVariables();

        // Device properties
        String avdName = expandVariables(envVars, buildVars, this.avdName);
        String osVersion = expandVariables(envVars, buildVars, this.osVersion);
        String screenDensity = expandVariables(envVars, buildVars, this.screenDensity);
        String screenResolution = expandVariables(envVars, buildVars, this.screenResolution);
        String deviceLocale = expandVariables(envVars, buildVars, this.deviceLocale);
        String sdCardSize = expandVariables(envVars, buildVars, this.sdCardSize);

        // SDK location
        String androidHome = expandVariables(envVars, buildVars, descriptor.androidHome);
        androidHome = validateAndroidHome(launcher, localVars, androidHome);

        // Despite the nice inline checks and warnings when the user is editing the config,
        // these are not binding, so the user may have saved invalid configuration.
        // Here we check whether or not it's worth proceeding based on the saved values.
        String configError = isConfigValid(avdName, osVersion, screenDensity, screenResolution,
                deviceLocale, sdCardSize);
        if (configError != null) {
            log(logger, Messages.ERROR_MISCONFIGURED(configError));
            build.setResult(Result.NOT_BUILT);
            return null;
        }

        // Confirm that tools are available on PATH
        if (!validateAndroidToolsInPath(launcher, androidHome)) {
            log(logger, Messages.SDK_TOOLS_NOT_FOUND());
            build.setResult(Result.NOT_BUILT);
            return null;
        }

        // Ok, everything looks good.. let's go
        String displayHome = androidHome == null ? Messages.USING_PATH() : androidHome;
        log(logger, Messages.USING_SDK(displayHome));
        EmulatorConfig emuConfig = EmulatorConfig.create(avdName, osVersion, screenDensity,
                screenResolution, deviceLocale, sdCardSize, wipeData, showWindow);

        return doSetUp(build, launcher, listener, androidHome, emuConfig);
    }

    private Environment doSetUp(final AbstractBuild build, final Launcher launcher,
            final BuildListener listener, final String androidHome, final EmulatorConfig emuConfig)
                throws IOException, InterruptedException {
        final PrintStream logger = listener.getLogger();

        // First ensure that emulator exists
        final Computer computer = Computer.currentComputer();
        final boolean emulatorAlreadyExists;
        try {
            Callable task = emuConfig.getEmulatorCreationTask(androidHome, launcher.isUnix(), listener);
            emulatorAlreadyExists = launcher.getChannel().call(task);
        } catch (EmulatorDiscoveryException ex) {
            log(logger, Messages.CANNOT_START_EMULATOR(ex.getMessage()));
            build.setResult(Result.FAILURE);
            return null;
        } catch (AndroidEmulatorException ex) {
            log(logger, Messages.COULD_NOT_CREATE_EMULATOR(ex.getMessage()));
            build.setResult(Result.NOT_BUILT);
            return null;
        }

        // Delay start up by the configured amount of time
        final int delaySecs = getStartupDelay();
        if (delaySecs > 0) {
            log(logger, Messages.DELAYING_START_UP(delaySecs));
            Thread.sleep(delaySecs * 1000);
        }

        // Use the Port Allocator plugin to reserve the two ports we need
        final PortAllocationManager portAllocator = PortAllocationManager.getManager(computer);
        final int userPort = portAllocator.allocateRandom(build, 0);
        final int adbPort = portAllocator.allocateRandom(build, 0);

        // Compile complete command for starting emulator
        final String avdArgs = emuConfig.getCommandArguments();
        String emulatorArgs = String.format("-ports %s,%s %s", userPort, adbPort, avdArgs);
        ArgumentListBuilder emulatorCmd = Utils.getToolCommand(launcher, androidHome, "emulator", "emulator.exe", emulatorArgs);

        // Start emulator process
        log(logger, Messages.STARTING_EMULATOR());
        if (emulatorAlreadyExists && emuConfig.shouldWipeData()) {
            log(logger, Messages.ERASING_EXISTING_EMULATOR_DATA());
        }
        final long bootTime = System.currentTimeMillis();
        final EnvVars buildEnvironment = build.getEnvironment(TaskListener.NULL);
        final ProcStarter procStarter = launcher.launch().stdout(logger).stderr(logger);
        final Proc emulatorProcess = procStarter.envs(buildEnvironment).cmds(emulatorCmd).start();

        // Wait for TCP socket to become available
        boolean socket = waitForSocket(launcher, adbPort, ADB_CONNECT_TIMEOUT_MS);
        if (!socket || !emulatorProcess.isAlive()) {
            log(logger, Messages.EMULATOR_DID_NOT_START());
            build.setResult(Result.NOT_BUILT);
            cleanUp(logger, portAllocator, emulatorProcess, adbPort, userPort);
            return null;
        }

        // Notify adb of our existence
        final String adbConnectArgs = "connect localhost:"+ adbPort;
        ArgumentListBuilder adbConnectCmd = Utils.getToolCommand(launcher, androidHome, "adb", "adb.exe", adbConnectArgs);
        int result = procStarter.cmds(adbConnectCmd).stdout(new NullOutputStream()).start().join();
        if (result != 0) { // adb currently only ever returns 0!
            log(logger, Messages.CANNOT_CONNECT_TO_EMULATOR());
            build.setResult(Result.NOT_BUILT);
            cleanUp(logger, portAllocator, emulatorProcess, adbPort, userPort);
            return null;
        }

        // Start dumping logs to disk
        final File artifactsDir = build.getArtifactsDir();
        final FilePath logcatFile = build.getWorkspace().createTempFile("logcat_", ".log");
        final OutputStream logcatStream = logcatFile.write();
        final String logcatArgs = "-s localhost:"+ adbPort +" logcat -v time";
        ArgumentListBuilder logcatCmd = Utils.getToolCommand(launcher, androidHome, "adb", "adb.exe", logcatArgs);
        final Proc logWriter = procStarter.cmds(logcatCmd).stdout(logcatStream).stderr(new NullOutputStream()).start();

        // Monitor device for boot completion signal
        log(logger, Messages.WAITING_FOR_BOOT_COMPLETION());
        int bootTimeout = BOOT_COMPLETE_TIMEOUT_MS;
        if (!emulatorAlreadyExists || emuConfig.shouldWipeData()) {
            bootTimeout *= 4;
        }
        boolean bootSucceeded = waitForBootCompletion(logger, launcher, androidHome, adbPort, bootTimeout);
        if (!bootSucceeded) {
            log(logger, Messages.BOOT_COMPLETION_TIMED_OUT(bootTimeout / 1000));
            build.setResult(Result.NOT_BUILT);
            cleanUp(logger, launcher, androidHome, portAllocator, emulatorProcess,
                    adbPort, userPort, logWriter, logcatFile, logcatStream, artifactsDir);
            return null;
        }
        final long bootCompleteTime = System.currentTimeMillis();
        log(logger, Messages.EMULATOR_IS_READY((bootCompleteTime - bootTime) / 1000));

        // Return wrapped environment
        return new Environment() {
            @Override
            public void buildEnvVars(Map env) {
                env.put("ANDROID_AVD_DEVICE", "localhost:"+ adbPort);
                env.put("ANDROID_AVD_ADB_PORT", Integer.toString(adbPort));
                env.put("ANDROID_AVD_USER_PORT", Integer.toString(userPort));
                env.put("ANDROID_AVD_NAME", emuConfig.getAvdName());
                if (!emuConfig.isNamedEmulator()) {
                    env.put("ANDROID_AVD_OS", emuConfig.getOsVersion().toString());
                    env.put("ANDROID_AVD_DENSITY", emuConfig.getScreenDensity().toString());
                    env.put("ANDROID_AVD_RESOLUTION", emuConfig.getScreenResolution().toString());
                    env.put("ANDROID_AVD_SKIN", emuConfig.getScreenResolution().getSkinName());
                    env.put("ANDROID_AVD_LOCALE", emuConfig.getDeviceLocale());
                }
            }

            @Override
            @SuppressWarnings("unchecked")
            public boolean tearDown(AbstractBuild build, BuildListener listener)
                    throws IOException, InterruptedException {
                cleanUp(logger, launcher, androidHome, portAllocator, emulatorProcess,
                        adbPort, userPort, logWriter, logcatFile, logcatStream, artifactsDir);

                return true;
            }
        };
    }

    /** Helper method for writing to the build log in a consistent manner. */
    synchronized static void log(final PrintStream logger, final String message) {
        log(logger, message, false);
    }

    /** Helper method for writing to the build log in a consistent manner. */
    synchronized static void log(final PrintStream logger, String message, boolean indent) {
        if (indent) {
            message = '\t' + message.replace("\n", "\n\t");
        } else {
            logger.print("[android] ");
        }
        logger.println(message);
    }

    /**
     * Called when this wrapper needs to exit, so we need to clean up some processes etc.
     *
     * @param logger The build logger.
     * @param portAllocator The port allocator used.
     * @param emulatorProcess The Android emulator process.
     * @param adbPort The ADB port used by the emulator.
     * @param userPort The user port used by the emulator.
     */
    private void cleanUp(PrintStream logger, PortAllocationManager portAllocator,
            Proc emulatorProcess, int adbPort, int userPort) throws IOException, InterruptedException {
        cleanUp(logger, null, null, portAllocator, emulatorProcess, adbPort, userPort, null, null, null, null);
    }

    /**
     * Called when this wrapper needs to exit, so we need to clean up some processes etc.
     *
     * @param logger The build logger.
     * @param launcher The launcher for the remote node.
     * @param androidHome The Android SDK root.
     * @param portAllocator The port allocator used.
     * @param emulatorProcess The Android emulator process.
     * @param adbPort The ADB port used by the emulator.
     * @param userPort The user port used by the emulator.
     * @param logcatProcess The adb logcat process.
     * @param logcatFile The file the logcat output is being written to.
     * @param logcatStream The stream the logcat output is being written to.
     * @param artifactsDir The directory where build artifacts should go.
     */
    private void cleanUp(PrintStream logger, Launcher launcher, String androidHome,
            PortAllocationManager portAllocator, Proc emulatorProcess, int adbPort,
            int userPort, Proc logcatProcess, FilePath logcatFile,
            OutputStream logcatStream, File artifactsDir)
                throws IOException, InterruptedException {
        // FIXME: Sometimes on Windows neither the emulator.exe nor the adb.exe processes die.
        //        Launcher.kill(EnvVars) does not appear to help either.
        //        This is (a) inconsistent; (b) very annoying.

        // Disconnect emulator from adb, if it's running
        if (launcher != null) {
            final String args = "disconnect localhost:"+ adbPort;
            ArgumentListBuilder adbDisconnectCmd = Utils.getToolCommand(launcher, androidHome, "adb", "adb.exe", args);
            final ProcStarter procStarter = launcher.launch().stdout(logger).stderr(logger);
            procStarter.cmds(adbDisconnectCmd).stdout(new NullOutputStream()).start().join();
        }

        // Stop emulator process and free up TCP ports
        log(logger, Messages.STOPPING_EMULATOR());
        emulatorProcess.kill();
        portAllocator.free(adbPort);
        portAllocator.free(userPort);

        // Archive the logs
        if (logcatProcess != null) {
            logcatProcess.kill();
            logcatStream.close();
            if (logcatFile.length() != 0) {
                log(logger, Messages.ARCHIVING_LOG());
                logcatFile.copyTo(new FilePath(artifactsDir).child("logcat.txt"));
            }
            logcatFile.delete();
        }
    }

    /**
     * Expands the variable in the given string to its value in the environment variables available
     * to this build.  The Hudson-specific build variables for this build are then substituted.
     *
     * @param envVars  Map of the environment variables.
     * @param buildVars  Map of the build-specific variables.
     * @param token  The token which may or may not contain variables in the format ${foo}.
     * @return  The given token, with applicable variable expansions done.
     */
    private String expandVariables(EnvVars envVars, Map buildVars,
            String token) {
        String result = Util.fixEmptyAndTrim(token);
        if (result != null) {
            result = Util.replaceMacro(Util.replaceMacro(result, envVars), buildVars);
        }
        return result;
    }

    /**
     * Validates this instance's configuration.
     *
     * @return A human-readable error message, or null if the config is valid.
     */
    private String isConfigValid(String avdName, String osVersion, String screenDensity,
            String screenResolution, String deviceLocale, String sdCardSize) {
        if (getUseNamedEmulator()) {
            ValidationResult result = descriptor.doCheckAvdName(avdName, false);
            if (result.isFatal()) {
                return result.getMessage();
            }
        } else {
            ValidationResult result = descriptor.doCheckOsVersion(osVersion, false);
            if (result.isFatal()) {
                return result.getMessage();
            }
            result = descriptor.doCheckScreenDensity(screenDensity, false);
            if (result.isFatal()) {
                return result.getMessage();
            }
            result = descriptor.doCheckScreenResolution(screenResolution, null, false);
            if (result.isFatal()) {
                return result.getMessage();
            }
            result = descriptor.doCheckDeviceLocale(deviceLocale, false);
            if (result.isFatal()) {
                return result.getMessage();
            }
            result = descriptor.doCheckSdCardSize(sdCardSize, false);
            if (result.isFatal()) {
                return result.getMessage();
            }
        }

        return null;
    }

    /**
     * Tries to validate the given Android SDK root directory; otherwise tries to
     * locate a copy of the SDK by checking for common environment variables.
     *
     * @param launcher The launcher for the remote node.
     * @param envVars Environment variables for the build.
     * @param androidHome The (variable-expanded) SDK root given in global config.
     * @return Either a discovered SDK path or, if all else fails, the given androidHome value.
     */
    private String validateAndroidHome(final Launcher launcher, final EnvVars envVars,
            final String androidHome) {
        Callable task = new Callable() {
            public String call() throws InterruptedException {
                // Verify existence of provided value
                if (validateHomeDir(androidHome)) {
                    return androidHome;
                }

                // Check for common environment variables
                String[] keys = { "ANDROID_SDK_ROOT", "ANDROID_SDK_HOME",
                                  "ANDROID_HOME", "ANDROID_SDK" };
                for (String key : keys) {
                    String home = envVars.get(key);
                    if (validateHomeDir(home)) {
                        return home;
                    }
                }

                // If all else fails, return what we were given
                return androidHome;
            }

            private boolean validateHomeDir(String dir) {
                if (Util.fixEmptyAndTrim(dir) == null) {
                    return false;
                }
                return !descriptor.doCheckAndroidHome(new File(dir), false).isFatal();
            }

            private static final long serialVersionUID = 1L;
        };

        String result = androidHome;
        try {
            result = launcher.getChannel().call(task);
        } catch (InterruptedException e) {
            // Ignore; will return default value
        } catch (IOException e) {
            // Ignore; will return default value
        }
        return result;
    }

    /**
     * Validates whether the required SDK tools can be reached, either from the given root or PATH.
     *
     * @param launcher The launcher for the remote node.
     * @param androidHome The (variable-expanded) SDK root given in global config.
     * @return true if all the required tools are available.
     */
    private boolean validateAndroidToolsInPath(Launcher launcher, final String androidHome) {
        final String executable = "tools/" + (launcher.isUnix() ? "adb" : "adb.exe");

        Callable task = new Callable() {
            public Boolean call() throws IOException {
                String sep = System.getProperty("path.separator");
                List list = Arrays.asList(System.getenv("PATH").split(sep));
                List paths = new ArrayList(list);
                paths.add(0, androidHome);
                for (String path : paths) {
                    if (new File(path, executable).exists()) {
                        return true;
                    }
                }

                return false;
            }
            private static final long serialVersionUID = 1L;
        };

        try {
            return launcher.getChannel().call(task);
        } catch (IOException e) {
            // Ignore
        } catch (InterruptedException e) {
            // Ignore
        }
        return false;
    }

    /**
     * Waits for a socket on the remote machine's localhost to become available, or times out.
     *
     * @param launcher The launcher for the remote node.
     * @param port The port to try and connect to.
     * @param timeout How long to keep trying (in milliseconds) before giving up.
     * @return true if the socket was available, false if we timed-out.
     */
    private boolean waitForSocket(Launcher launcher, int port, int timeout) {
        try {
            LocalPortOpenTask task = new LocalPortOpenTask(port, timeout);
            return launcher.getChannel().call(task);
        } catch (InterruptedException ex) {
            // Ignore
        } catch (IOException e) {
            // Ignore
        }

        return false;
    }

    /**
     * Checks whether the emulator running on the given port has finished booting yet, or times out.
     *
     * @param logger The build logger.
     * @param launcher The launcher for the remote node.
     * @param androidHome The Android SDK root.
     * @param port The emulator's ADB port.
     * @param timeout How long to keep trying (in milliseconds) before giving up.
     * @return true if the emulator has booted, false if we timed-out.
     */
    private boolean waitForBootCompletion(final PrintStream logger, final Launcher launcher,
            final String androidHome, final int port, final int timeout) {
        long start = System.currentTimeMillis();
        int sleep = timeout / (int) Math.sqrt(timeout / 1000);

        final String serialNo = "localhost:"+ port;
        final String args = "-s "+ serialNo +" shell getprop dev.bootcomplete";
        ArgumentListBuilder cmd = Utils.getToolCommand(launcher, androidHome, "adb", "adb.exe", args);

        try {
            while (System.currentTimeMillis() < start + timeout) {
                ByteArrayOutputStream stream = new ByteArrayOutputStream(4);

                // Run "getprop"
                launcher.launch().cmds(cmd).stdout(stream).start().join();

                // Check output
                String result = stream.toString().trim();
                if (result.equals("1")) {
                    return true;
                }

                // Otherwise continue...
                Thread.sleep(sleep);
            }
        } catch (InterruptedException ex) {
            log(logger, Messages.INTERRUPTED_DURING_BOOT_COMPLETION());
        } catch (IOException ex) {
            log(logger, Messages.COULD_NOT_CHECK_BOOT_COMPLETION());
            ex.printStackTrace(logger);
        }

        return false;
    }

    @Extension(ordinal=-100) // Negative ordinal makes us execute after other wrappers (i.e. Xvnc)
    public static final class DescriptorImpl extends BuildWrapperDescriptor implements Serializable {

        private static final long serialVersionUID = 1L;

        // From hudson.Util.VARIABLE
        private static final String VARIABLE_REGEX = "\\$([A-Za-z0-9_]+|\\{[A-Za-z0-9_]+\\}|\\$)";

        /**
         * The Android SDK home directory.  Can include variables, e.g. ${ANDROID_HOME}.
         * 

If null, we will just assume the required commands are on the PATH.

*/ public String androidHome; public DescriptorImpl() { super(AndroidEmulator.class); load(); } @Override public String getDisplayName() { return Messages.JOB_DESCRIPTION(); } @Override public boolean configure(StaplerRequest req, JSONObject json) throws FormException { req.bindParameters(this, "android-emulator."); save(); return true; } @Override public BuildWrapper newInstance(StaplerRequest req, JSONObject formData) throws FormException { String avdName = null; String osVersion = null; String screenDensity = null; String screenResolution = null; String deviceLocale = null; String sdCardSize = null; boolean wipeData = false; boolean showWindow = true; int startupDelay = 0; JSONObject emulatorData = formData.getJSONObject("useNamed"); String useNamedValue = emulatorData.getString("value"); if (Boolean.parseBoolean(useNamedValue)) { avdName = Util.fixEmptyAndTrim(emulatorData.getString("avdName")); } else { osVersion = Util.fixEmptyAndTrim(emulatorData.getString("osVersion")); screenDensity = Util.fixEmptyAndTrim(emulatorData.getString("screenDensity")); screenResolution = Util.fixEmptyAndTrim(emulatorData.getString("screenResolution")); deviceLocale = Util.fixEmptyAndTrim(emulatorData.getString("deviceLocale")); sdCardSize = Util.fixEmptyAndTrim(emulatorData.getString("sdCardSize")); if (sdCardSize != null) { sdCardSize = sdCardSize.toUpperCase().replaceAll("[ B]", ""); } } wipeData = formData.getBoolean("wipeData"); showWindow = formData.getBoolean("showWindow"); try { startupDelay = Integer.parseInt(formData.getString("startupDelay")); } catch (NumberFormatException e) {} return new AndroidEmulator(avdName, osVersion, screenDensity, screenResolution, deviceLocale, sdCardSize, wipeData, showWindow, startupDelay); } @Override public String getHelpFile() { return "/plugin/android-emulator/help-buildConfig.html"; } @Override public boolean isApplicable(AbstractProject item) { return true; } /** Used in config.jelly: Lists the OS versions available. */ public AndroidPlatform[] getAndroidVersions() { return AndroidPlatform.PRESETS; } /** Used in config.jelly: Lists the screen densities available. */ public ScreenDensity[] getDeviceDensities() { return ScreenDensity.PRESETS; } /** Used in config.jelly: Lists the screen resolutions available. */ public ScreenResolution[] getDeviceResolutions() { return ScreenResolution.PRESETS; } /** Used in config.jelly: Lists the locales available. */ public String[] getEmulatorLocales() { return Constants.EMULATOR_LOCALES; } public FormValidation doCheckAvdName(@QueryParameter String value) { return doCheckAvdName(value, true).getFormValidation(); } private ValidationResult doCheckAvdName(String avdName, boolean allowVariables) { if (avdName == null || avdName.equals("")) { return ValidationResult.error(Messages.AVD_NAME_REQUIRED()); } String regex = Constants.REGEX_AVD_NAME; if (allowVariables) { regex = "(("+ Constants.REGEX_AVD_NAME +")*("+ VARIABLE_REGEX +")*)+"; } if (!avdName.matches(regex)) { return ValidationResult.error(Messages.INVALID_AVD_NAME()); } return ValidationResult.ok(); } public FormValidation doCheckOsVersion(@QueryParameter String value) { return doCheckOsVersion(value, true).getFormValidation(); } private ValidationResult doCheckOsVersion(String osVersion, boolean allowVariables) { if (osVersion == null || osVersion.equals("")) { return ValidationResult.error(Messages.OS_VERSION_REQUIRED()); } if (!allowVariables && osVersion.matches(VARIABLE_REGEX)) { return ValidationResult.error(Messages.INVALID_OS_VERSION()); } return ValidationResult.ok(); } public FormValidation doCheckScreenDensity(@QueryParameter String value) { return doCheckScreenDensity(value, true).getFormValidation(); } private ValidationResult doCheckScreenDensity(String density, boolean allowVariables) { if (density == null || density.equals("")) { return ValidationResult.error(Messages.SCREEN_DENSITY_REQUIRED()); } String regex = Constants.REGEX_SCREEN_DENSITY; if (allowVariables) { regex += "|"+ VARIABLE_REGEX; } if (!density.matches(regex)) { return ValidationResult.error(Messages.SCREEN_DENSITY_NOT_NUMERIC()); } return ValidationResult.ok(); } public FormValidation doCheckScreenResolution(@QueryParameter String value, @QueryParameter String density) { return doCheckScreenResolution(value, density, true).getFormValidation(); } private ValidationResult doCheckScreenResolution(String resolution, String density, boolean allowVariables) { if (resolution == null || resolution.equals("")) { return ValidationResult.error(Messages.SCREEN_RESOLUTION_REQUIRED()); } String regex = Constants.REGEX_SCREEN_RESOLUTION_FULL; if (allowVariables) { regex += "|"+ VARIABLE_REGEX; } if (!resolution.matches(regex)) { return ValidationResult.error(Messages.INVALID_RESOLUTION_FORMAT()); } // Check for shenanigans ScreenResolution resolutionValue = ScreenResolution.valueOf(resolution); ScreenDensity densityValue = ScreenDensity.valueOf(density); if (resolutionValue != null && densityValue != null && !resolutionValue.isCustomResolution() && !densityValue.isCustomDensity()) { boolean densityFound = false; for (ScreenDensity okDensity : resolutionValue.getApplicableDensities()) { if (okDensity.equals(densityValue)) { densityFound = true; break; } } if (!densityFound) { return ValidationResult.warning(Messages.SUSPECT_RESOLUTION(resolution, densityValue)); } } return ValidationResult.ok(); } public FormValidation doCheckDeviceLocale(@QueryParameter String value) { return doCheckDeviceLocale(value, true).getFormValidation(); } private ValidationResult doCheckDeviceLocale(String locale, boolean allowVariables) { if (locale == null || locale.equals("")) { return ValidationResult.warning(Messages.DEFAULT_LOCALE_WARNING(Constants.DEFAULT_LOCALE)); } String regex = Constants.REGEX_LOCALE; if (allowVariables) { regex += "|"+ VARIABLE_REGEX; } if (!locale.matches(regex)) { return ValidationResult.error(Messages.LOCALE_FORMAT_WARNING()); } return ValidationResult.ok(); } public FormValidation doCheckSdCardSize(@QueryParameter String value) { return doCheckSdCardSize(value, true).getFormValidation(); } private ValidationResult doCheckSdCardSize(String sdCardSize, boolean allowVariables) { if (sdCardSize == null || sdCardSize.equals("")) { // No value, no SD card is created return ValidationResult.ok(); } String regex = Constants.REGEX_SD_CARD_SIZE; if (allowVariables) { regex += "|"+ VARIABLE_REGEX; } if (!sdCardSize.matches(regex)) { return ValidationResult.error(Messages.INVALID_SD_CARD_SIZE()); } // Validate size of SD card: New AVD requires at least 9MB Matcher matcher = Pattern.compile(Constants.REGEX_SD_CARD_SIZE).matcher(sdCardSize); if (matcher.matches()) { long bytes = Long.parseLong(matcher.group(1)); if (matcher.group(2).toUpperCase().equals("M")) { // Convert to KB bytes *= 1024; } bytes *= 1024L; if (bytes < (9 * 1024 * 1024)) { return ValidationResult.error(Messages.SD_CARD_SIZE_TOO_SMALL()); } } return ValidationResult.ok(); } public FormValidation doCheckAndroidHome(@QueryParameter File value) { return doCheckAndroidHome(value, true).getFormValidation(); } private ValidationResult doCheckAndroidHome(File sdkRoot, boolean fromWebConfig) { // This can be used to check the existence of a file on the server, so needs to be protected if (fromWebConfig && !Hudson.getInstance().hasPermission(Hudson.ADMINISTER)) { return ValidationResult.ok(); } // Check the utter basics if (fromWebConfig && (sdkRoot == null || sdkRoot.getPath().equals(""))) { return ValidationResult.ok(); } if (!sdkRoot.isDirectory()) { if (fromWebConfig && sdkRoot.getPath().matches(".*("+ VARIABLE_REGEX +").*")) { return ValidationResult.ok(); } return ValidationResult.error(Messages.INVALID_DIRECTORY()); } // We'll be using items from the tools and platforms directories for (String dirName : new String[] { "tools", "platforms" }) { File dir = new File(sdkRoot, dirName); if (!dir.exists() || !dir.isDirectory()) { return ValidationResult.error(Messages.INVALID_SDK_DIRECTORY()); } } // So long as the basic executables exist, we're happy int toolsFound = 0; final String[] requiredTools = { "adb", "android", "emulator" }; for (String toolName : requiredTools) { for (String extension : new String[] { "", ".bat", ".exe" }) { File tool = new File(sdkRoot, "tools/"+ toolName + extension); if (tool.exists() && tool.isFile()) { toolsFound++; break; } } } if (toolsFound != requiredTools.length) { return ValidationResult.errorWithMarkup(Messages.REQUIRED_SDK_TOOLS_NOT_FOUND()); } // Give the user a nice warning (not error) if they've not downloaded any platforms yet File platformsDir = new File(sdkRoot, "platforms"); if (platformsDir.list().length == 0) { return ValidationResult.warning(Messages.SDK_PLATFORMS_EMPTY()); } return ValidationResult.ok(); } } /** Task that will block until it can either connect to a port on localhost, or it times-out. */ private static final class LocalPortOpenTask implements Callable { private static final long serialVersionUID = 1L; private final int port; private final int timeout; /** * @param port The local TCP port to attempt to connect to. * @param timeout How long to keep trying (in milliseconds) before giving up. */ public LocalPortOpenTask(int port, int timeout) { this.port = port; this.timeout = timeout; } public Boolean call() throws InterruptedException { final long start = System.currentTimeMillis(); while (System.currentTimeMillis() < start + timeout) { try { Socket socket = new Socket("127.0.0.1", port); socket.getOutputStream(); socket.close(); return true; } catch (IOException ex) { // Ignore } Thread.sleep(1000); } return false; } } /** The Java equivalent of /dev/null. */ private static final class NullOutputStream extends OutputStream { @Override public void write(int b) throws IOException { // La la la } @Override public void write(byte[] b) throws IOException { // I can't hear you } @Override public void write(byte[] b, int off, int len) throws IOException { // Nope, still can't hear you } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy