| /******************************************************************************* |
| * Copyright (c) 2008, 2010 IBM Corporation and others. |
| * All rights reserved. This program and the accompanying materials |
| * are made available under the terms of the Eclipse Public License v1.0 |
| * which accompanies this distribution, and is available at |
| * http://www.eclipse.org/legal/epl-v10.html |
| * |
| * Contributors: |
| * IBM Corporation - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.equinox.internal.p2.ui.sdk.scheduler; |
| |
| import com.ibm.icu.util.Calendar; |
| import com.ibm.icu.util.ULocale; |
| import org.eclipse.core.runtime.*; |
| import org.eclipse.equinox.internal.p2.core.helpers.ServiceHelper; |
| import org.eclipse.equinox.internal.provisional.p2.updatechecker.*; |
| import org.eclipse.equinox.p2.core.IProvisioningAgent; |
| import org.eclipse.equinox.p2.engine.IProfile; |
| import org.eclipse.equinox.p2.engine.IProfileRegistry; |
| import org.eclipse.equinox.p2.engine.query.IUProfilePropertyQuery; |
| import org.eclipse.equinox.p2.metadata.IInstallableUnit; |
| import org.eclipse.equinox.p2.query.IQuery; |
| import org.eclipse.jface.preference.IPreferenceStore; |
| import org.eclipse.ui.IStartup; |
| import org.eclipse.ui.statushandlers.StatusManager; |
| |
| /** |
| * This plug-in is loaded on startup to register with the update checker. |
| * |
| * @since 3.5 |
| */ |
| public class AutomaticUpdateScheduler implements IStartup { |
| // values are to be picked up from the arrays DAYS and HOURS |
| public static final String P_DAY = "day"; //$NON-NLS-1$ |
| |
| public static final String P_HOUR = "hour"; //$NON-NLS-1$ |
| |
| public static final String[] DAYS; |
| |
| public static final String[] HOURS = {AutomaticUpdateMessages.SchedulerStartup_1AM, AutomaticUpdateMessages.SchedulerStartup_2AM, AutomaticUpdateMessages.SchedulerStartup_3AM, AutomaticUpdateMessages.SchedulerStartup_4AM, AutomaticUpdateMessages.SchedulerStartup_5AM, AutomaticUpdateMessages.SchedulerStartup_6AM, AutomaticUpdateMessages.SchedulerStartup_7AM, AutomaticUpdateMessages.SchedulerStartup_8AM, AutomaticUpdateMessages.SchedulerStartup_9AM, AutomaticUpdateMessages.SchedulerStartup_10AM, AutomaticUpdateMessages.SchedulerStartup_11AM, AutomaticUpdateMessages.SchedulerStartup_12PM, AutomaticUpdateMessages.SchedulerStartup_1PM, AutomaticUpdateMessages.SchedulerStartup_2PM, AutomaticUpdateMessages.SchedulerStartup_3PM, AutomaticUpdateMessages.SchedulerStartup_4PM, |
| AutomaticUpdateMessages.SchedulerStartup_5PM, AutomaticUpdateMessages.SchedulerStartup_6PM, AutomaticUpdateMessages.SchedulerStartup_7PM, AutomaticUpdateMessages.SchedulerStartup_8PM, AutomaticUpdateMessages.SchedulerStartup_9PM, AutomaticUpdateMessages.SchedulerStartup_10PM, AutomaticUpdateMessages.SchedulerStartup_11PM, AutomaticUpdateMessages.SchedulerStartup_12AM,}; |
| |
| private IUpdateListener listener = null; |
| private IUpdateChecker checker = null; |
| String profileId; |
| |
| static { |
| Calendar calendar = Calendar.getInstance(new ULocale(Platform.getNL())); |
| String[] daysAsStrings = {AutomaticUpdateMessages.SchedulerStartup_day, AutomaticUpdateMessages.SchedulerStartup_Sunday, AutomaticUpdateMessages.SchedulerStartup_Monday, AutomaticUpdateMessages.SchedulerStartup_Tuesday, AutomaticUpdateMessages.SchedulerStartup_Wednesday, AutomaticUpdateMessages.SchedulerStartup_Thursday, AutomaticUpdateMessages.SchedulerStartup_Friday, AutomaticUpdateMessages.SchedulerStartup_Saturday}; |
| int firstDay = calendar.getFirstDayOfWeek(); |
| DAYS = new String[8]; |
| DAYS[0] = daysAsStrings[0]; |
| int countDays = 0; |
| for (int i = firstDay; i <= 7; i++) { |
| DAYS[++countDays] = daysAsStrings[i]; |
| } |
| for (int i = 1; i < firstDay; i++) { |
| DAYS[++countDays] = daysAsStrings[i]; |
| } |
| } |
| |
| /** |
| * The constructor. |
| */ |
| public AutomaticUpdateScheduler() { |
| AutomaticUpdatePlugin.getDefault().setScheduler(this); |
| IProvisioningAgent agent = (IProvisioningAgent) ServiceHelper.getService(AutomaticUpdatePlugin.getContext(), IProvisioningAgent.SERVICE_NAME); |
| checker = (IUpdateChecker) agent.getService(IUpdateChecker.SERVICE_NAME); |
| if (checker == null) { |
| // Something did not initialize properly |
| IStatus status = new Status(IStatus.ERROR, AutomaticUpdatePlugin.PLUGIN_ID, AutomaticUpdateMessages.AutomaticUpdateScheduler_UpdateNotInitialized); |
| StatusManager.getManager().handle(status, StatusManager.LOG); |
| return; |
| } |
| profileId = IProfileRegistry.SELF; |
| } |
| |
| public void earlyStartup() { |
| scheduleUpdate(); |
| } |
| |
| public void shutdown() { |
| removeUpdateListener(); |
| } |
| |
| public void rescheduleUpdate() { |
| removeUpdateListener(); |
| IPreferenceStore pref = AutomaticUpdatePlugin.getDefault().getPreferenceStore(); |
| String schedule = pref.getString(PreferenceConstants.PREF_AUTO_UPDATE_SCHEDULE); |
| // See if we have a scheduled check or startup only. If it is |
| // startup only, there is nothing more to do now, a listener will |
| // be created on the next startup. |
| if (schedule.equals(PreferenceConstants.PREF_UPDATE_ON_STARTUP)) { |
| return; |
| } |
| scheduleUpdate(); |
| } |
| |
| private void scheduleUpdate() { |
| // Nothing to do if we don't know what profile we are checking |
| if (profileId == null) |
| return; |
| IPreferenceStore pref = AutomaticUpdatePlugin.getDefault().getPreferenceStore(); |
| // See if automatic search is enabled at all |
| if (!pref.getBoolean(PreferenceConstants.PREF_AUTO_UPDATE_ENABLED)) |
| return; |
| String schedule = pref.getString(PreferenceConstants.PREF_AUTO_UPDATE_SCHEDULE); |
| long delay = IUpdateChecker.ONE_TIME_CHECK; |
| long poll = IUpdateChecker.ONE_TIME_CHECK; |
| if (!schedule.equals(PreferenceConstants.PREF_UPDATE_ON_STARTUP)) { |
| delay = computeDelay(pref); |
| poll = computePoll(pref); |
| } |
| // We do not access the AutomaticUpdater directly when we register |
| // the listener. This prevents the UI classes from being started up |
| // too soon. |
| // see https://bugs.eclipse.org/bugs/show_bug.cgi?id=227582 |
| listener = new IUpdateListener() { |
| public void updatesAvailable(UpdateEvent event) { |
| AutomaticUpdatePlugin.getDefault().getAutomaticUpdater().updatesAvailable(event); |
| } |
| |
| }; |
| checker.addUpdateCheck(profileId, getProfileQuery(), delay, poll, listener); |
| |
| } |
| |
| private IQuery<IInstallableUnit> getProfileQuery() { |
| // We specifically avoid using the default policy's root property so that we don't load all the |
| // p2 UI classes in doing so. |
| return new IUProfilePropertyQuery(IProfile.PROP_PROFILE_ROOT_IU, Boolean.TRUE.toString()); |
| } |
| |
| private int getDay(IPreferenceStore pref) { |
| String day = pref.getString(P_DAY); |
| for (int d = 0; d < DAYS.length; d++) |
| if (DAYS[d].equals(day)) |
| switch (d) { |
| case 0 : |
| return -1; |
| case 1 : |
| return Calendar.MONDAY; |
| case 2 : |
| return Calendar.TUESDAY; |
| case 3 : |
| return Calendar.WEDNESDAY; |
| case 4 : |
| return Calendar.THURSDAY; |
| case 5 : |
| return Calendar.FRIDAY; |
| case 6 : |
| return Calendar.SATURDAY; |
| case 7 : |
| return Calendar.SUNDAY; |
| } |
| return -1; |
| } |
| |
| private int getHour(IPreferenceStore pref) { |
| String hour = pref.getString(P_HOUR); |
| for (int h = 0; h < HOURS.length; h++) |
| if (HOURS[h].equals(hour)) |
| return h + 1; |
| return 1; |
| } |
| |
| /* |
| * Computes the number of milliseconds from this moment to the next |
| * scheduled update check. If that moment has already passed, returns 0L (start |
| * immediately). |
| */ |
| private long computeDelay(IPreferenceStore pref) { |
| |
| int target_d = getDay(pref); |
| int target_h = getHour(pref); |
| |
| Calendar calendar = Calendar.getInstance(); |
| // may need to use the BootLoader locale |
| int current_d = calendar.get(Calendar.DAY_OF_WEEK); |
| // starts with SUNDAY |
| int current_h = calendar.get(Calendar.HOUR_OF_DAY); |
| int current_m = calendar.get(Calendar.MINUTE); |
| int current_s = calendar.get(Calendar.SECOND); |
| int current_ms = calendar.get(Calendar.MILLISECOND); |
| |
| long delay = 0L; // milliseconds |
| |
| if (target_d == -1) { |
| // Compute the delay for "every day at x o'clock" |
| // Is it now ? |
| if (target_h == current_h && current_m == 0 && current_s == 0) |
| return delay; |
| |
| int delta_h = target_h - current_h; |
| if (target_h <= current_h) |
| delta_h += 24; |
| delay = ((delta_h * 60 - current_m) * 60 - current_s) * 1000 - current_ms; |
| return delay; |
| } |
| // Compute the delay for "every Xday at x o'clock" |
| // Is it now ? |
| if (target_d == current_d && target_h == current_h && current_m == 0 && current_s == 0) |
| return delay; |
| |
| int delta_d = target_d - current_d; |
| if (target_d < current_d || target_d == current_d && (target_h < current_h || target_h == current_h && current_m > 0)) |
| delta_d += 7; |
| |
| delay = (((delta_d * 24 + target_h - current_h) * 60 - current_m) * 60 - current_s) * 1000 - current_ms; |
| return delay; |
| } |
| |
| /* |
| * Computes the number of milliseconds for the polling frequency. |
| * We have already established that there is a schedule, vs. only |
| * on startup. |
| */ |
| private long computePoll(IPreferenceStore pref) { |
| |
| int target_d = getDay(pref); |
| if (target_d == -1) { |
| // Every 24 hours |
| return 24 * 60 * 60 * 1000; |
| } |
| return 7 * 24 * 60 * 60 * 1000; |
| } |
| |
| private void removeUpdateListener() { |
| // Remove the current listener if there is one |
| if (listener != null && checker != null) { |
| checker.removeUpdateCheck(listener); |
| listener = null; |
| } |
| } |
| } |