Search in sources :

Example 1 with RegisteredResourceImpl

use of org.apache.sling.installer.core.impl.RegisteredResourceImpl in project sling by apache.

the class BundleTaskCreator method createTask.

/**
     * Create a bundle task - install, update or remove
     *
	 * @see org.apache.sling.installer.api.tasks.InstallTaskFactory#createTask(org.apache.sling.installer.api.tasks.TaskResourceGroup)
	 */
@Override
public InstallTask createTask(final TaskResourceGroup resourceList) {
    // quick check of the resource type.
    final TaskResource toActivate = resourceList.getActiveResource();
    if (toActivate.getType().equals(PersistentResourceList.RESTART_ACTIVE_BUNDLES_TYPE)) {
        return new RestartActiveBundlesTask(resourceList, this.taskSupport);
    }
    if (!toActivate.getType().equals(InstallableResource.TYPE_BUNDLE)) {
        return null;
    }
    // check if symbolic name and version is provided in the attributes
    if (toActivate.getAttribute(Constants.BUNDLE_SYMBOLICNAME) == null) {
        final Util.BundleHeaders headers = Util.readBundleHeaders(toActivate, logger);
        if (headers == null) {
            String message = MessageFormat.format("Resource of type bundle {0} is not really a bundle - manifest entries are missing.", toActivate);
            logger.info(message);
            return new ChangeStateTask(resourceList, ResourceState.IGNORED, message);
        }
        toActivate.setAttribute(Constants.BUNDLE_SYMBOLICNAME, headers.symbolicName);
        toActivate.setAttribute(Constants.BUNDLE_VERSION, headers.version);
        if (headers.activationPolicy != null) {
            toActivate.setAttribute(Constants.BUNDLE_ACTIVATIONPOLICY, headers.activationPolicy);
        }
    }
    final String symbolicName = (String) toActivate.getAttribute(Constants.BUNDLE_SYMBOLICNAME);
    final boolean isInstallerCoreBundle = this.bundleContext.getBundle().getSymbolicName().equals(symbolicName);
    // Uninstall
    final InstallTask result;
    if (toActivate.getState() == ResourceState.UNINSTALL) {
        // find the info with the exact version
        final BundleInfo info = this.getBundleInfo(symbolicName, (String) toActivate.getAttribute(Constants.BUNDLE_VERSION));
        // Remove corresponding bundle if present and if we installed it
        if (info != null) {
            // if this is an uninstall, check if we have to install an older version
            // in this case we should do an update instead of uninstall/install (!)
            Iterator<TaskResource> candidatesIt = ((EntityResourceList) resourceList).getActiveResourceIterator();
            TaskResource second = null;
            while (candidatesIt != null && second == null && candidatesIt.hasNext()) {
                TaskResource candidate = candidatesIt.next();
                boolean sameVersion = toActivate.getVersion().equals(candidate.getVersion());
                if (!sameVersion) {
                    if (bundleBlacklist.isBlacklisted(symbolicName, candidate.getVersion())) {
                        String message = MessageFormat.format("Uninstalling blacklisted bundle {0} found at {1}", symbolicName, candidate.getURL());
                        logger.info(message);
                        // blacklisted candidates should be uninstalled to no longer be taken into account anymore
                        ((RegisteredResourceImpl) candidate).setState(ResourceState.UNINSTALL, message);
                    } else {
                        second = candidate;
                    }
                }
            }
            if (second != null && (second.getState() == ResourceState.IGNORED || second.getState() == ResourceState.INSTALLED || second.getState() == ResourceState.INSTALL)) {
                second.setAttribute(FORCE_INSTALL_VERSION, info.version.toString());
                BundleUtil.clearBundleStart(second);
                logger.debug("Detected downgrade of bundle {}", symbolicName);
                result = new ChangeStateTask(resourceList, ResourceState.UNINSTALLED, null);
            } else {
                // prevent uninstalling the installer itself!
                if (isInstallerCoreBundle) {
                    logger.debug("Prevent completely uninstalling installer bundle {}", symbolicName);
                    result = new ChangeStateTask(resourceList, ResourceState.UNINSTALLED, null);
                } else {
                    result = new BundleRemoveTask(resourceList, this.taskSupport);
                }
            }
        } else {
            logger.debug("Bundle {}:{} is not installed anymore - nothing to remove.", symbolicName, toActivate.getAttribute(Constants.BUNDLE_VERSION));
            result = new ChangeStateTask(resourceList, ResourceState.UNINSTALLED, null);
        }
    // Install
    } else {
        // check for installer and system update
        final Integer asyncTaskCounter = (Integer) toActivate.getAttribute(InstallTask.ASYNC_ATTR_NAME);
        if (asyncTaskCounter != null) {
            if (isInstallerCoreBundle) {
                result = new InstallerBundleUpdateTask(resourceList, this.taskSupport);
            } else {
                // system bundle
                result = new ChangeStateTask(resourceList, ResourceState.INSTALLED, null, new String[] { InstallTask.ASYNC_ATTR_NAME }, null);
            }
        } else {
            final Version newVersion = new Version((String) toActivate.getAttribute(Constants.BUNDLE_VERSION));
            if (bundleBlacklist.isBlacklisted(symbolicName, newVersion)) {
                String message = MessageFormat.format("Ignoring blacklisted bundle {0} found at {1}", symbolicName, toActivate.getURL());
                logger.info(message);
                result = new ChangeStateTask(resourceList, ResourceState.IGNORED, message);
            } else {
                // for install and update, we want the bundle with the
                // highest version
                final BundleInfo info = this.getBundleInfo(symbolicName, null);
                // check if we should start the bundle as we installed it in the previous run
                if (info == null) {
                    // bundle is not installed yet: install
                    result = new BundleInstallTask(resourceList, this.taskSupport);
                } else if (BundleUtil.isBundleStart(toActivate)) {
                    result = new BundleStartTask(resourceList, info.id, this.taskSupport);
                } else {
                    boolean doUpdate = false;
                    final int compare = info.version.compareTo(newVersion);
                    if (compare < 0) {
                        // installed version is lower -> update
                        doUpdate = true;
                    } else if (compare > 0) {
                        final String forceVersion = (String) toActivate.getAttribute(FORCE_INSTALL_VERSION);
                        if (forceVersion != null && info.version.compareTo(new Version(forceVersion)) == 0) {
                            doUpdate = true;
                        } else {
                            logger.debug("Bundle " + info.symbolicName + " " + newVersion + " is not installed, bundle with higher version is already installed.");
                        }
                    } else if (compare == 0 && BundleInfo.isSnapshot(newVersion)) {
                        // installed, same version but SNAPSHOT
                        doUpdate = true;
                    }
                    if (doUpdate) {
                        logger.debug("Scheduling update of {}", toActivate);
                        // check if this is the system bundle
                        if (Constants.SYSTEM_BUNDLE_SYMBOLICNAME.equals(symbolicName)) {
                            result = new SystemBundleUpdateTask(resourceList, this.taskSupport);
                        // check if this is a installer update
                        } else if (isInstallerCoreBundle) {
                            result = new InstallerBundleUpdateTask(resourceList, this.taskSupport);
                        } else {
                            result = new BundleUpdateTask(resourceList, this.taskSupport);
                        }
                    } else if (compare == 0 && (isInstallerCoreBundle || Constants.SYSTEM_BUNDLE_SYMBOLICNAME.equals(symbolicName))) {
                        // the installer core bundle / system bundle has been updated, just set state
                        result = new ChangeStateTask(resourceList, ResourceState.INSTALLED, "This is the system bundle, therefore nothing was actually done!");
                    } else {
                        String message = MessageFormat.format("Nothing to install for {0}, same or newer version {1} already installed.", toActivate, newVersion);
                        logger.debug(message);
                        result = new ChangeStateTask(resourceList, ResourceState.IGNORED, message);
                    }
                }
            }
        }
        toActivate.setAttribute(FORCE_INSTALL_VERSION, null);
    }
    return result;
}
Also used : TaskResource(org.apache.sling.installer.api.tasks.TaskResource) Util(org.apache.sling.installer.core.impl.Util) InstallTask(org.apache.sling.installer.api.tasks.InstallTask) RegisteredResourceImpl(org.apache.sling.installer.core.impl.RegisteredResourceImpl) Version(org.osgi.framework.Version) ChangeStateTask(org.apache.sling.installer.api.tasks.ChangeStateTask) EntityResourceList(org.apache.sling.installer.core.impl.EntityResourceList)

Aggregations

ChangeStateTask (org.apache.sling.installer.api.tasks.ChangeStateTask)1 InstallTask (org.apache.sling.installer.api.tasks.InstallTask)1 TaskResource (org.apache.sling.installer.api.tasks.TaskResource)1 EntityResourceList (org.apache.sling.installer.core.impl.EntityResourceList)1 RegisteredResourceImpl (org.apache.sling.installer.core.impl.RegisteredResourceImpl)1 Util (org.apache.sling.installer.core.impl.Util)1 Version (org.osgi.framework.Version)1