package net.pms.update; import java.io.*; import java.util.Observable; import java.util.concurrent.Executor; import java.util.concurrent.Executors; import net.pms.PMS; import net.pms.configuration.PmsConfiguration; import net.pms.util.UriRetriever; import net.pms.util.UriRetrieverCallback; import net.pms.util.Version; import org.apache.commons.io.IOUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * Checks for and downloads new versions of PMS. * * @author Tim Cox (mail@tcox.org) */ public class AutoUpdater extends Observable implements UriRetrieverCallback { private static final String TARGET_FILENAME = "new-version.exe"; private static final Logger LOGGER = LoggerFactory.getLogger(AutoUpdater.class); private static final PmsConfiguration configuration = PMS.getConfiguration(); public static enum State { NOTHING_KNOWN, POLLING_SERVER, NO_UPDATE_AVAILABLE, UPDATE_AVAILABLE, DOWNLOAD_IN_PROGRESS, DOWNLOAD_FINISHED, EXECUTING_SETUP, ERROR } private final String serverUrl; private final UriRetriever uriRetriever = new UriRetriever(); private final AutoUpdaterServerProperties serverProperties = new AutoUpdaterServerProperties(); private final Version currentVersion; private Executor executor = Executors.newSingleThreadExecutor(); private State state = State.NOTHING_KNOWN; private Object stateLock = new Object(); private Throwable errorStateCause; private int bytesDownloaded = -1; private int totalBytes = -1; private boolean downloadCancelled = false; public AutoUpdater(String updateServerUrl, String currentVersion) { this.serverUrl = updateServerUrl; // may be null if updating is disabled this.currentVersion = new Version(currentVersion); } public void pollServer() { if (serverUrl != null) { // don't poll if the server URL is null executor.execute(new Runnable() { @Override public void run() { try { doPollServer(); } catch (UpdateException e) { setErrorState(e); } } }); } } private void doPollServer() throws UpdateException { assertNotInErrorState(); try { setState(State.POLLING_SERVER); byte[] propertiesAsData = uriRetriever.get(serverUrl); synchronized (stateLock) { serverProperties.loadFrom(propertiesAsData); setState(isUpdateAvailable() ? State.UPDATE_AVAILABLE : State.NO_UPDATE_AVAILABLE); } } catch (IOException e) { wrapException("Cannot download properties", e); } } public void getUpdateFromNetwork() { executor.execute(new Runnable() { @Override public void run() { try { doGetUpdateFromNetwork(); } catch (UpdateException e) { setErrorState(e); } } }); } public void runUpdateAndExit() { executor.execute(new Runnable() { @Override public void run() { try { doRunUpdateAndExit(); } catch (UpdateException e) { setErrorState(e); } } }); } private void setErrorState(UpdateException e) { synchronized (stateLock) { setState(State.ERROR); errorStateCause = e; } } private void doGetUpdateFromNetwork() throws UpdateException { assertNotInErrorState(); assertUpdateIsAvailable(); setState(State.DOWNLOAD_IN_PROGRESS); downloadUpdate(); setState(State.DOWNLOAD_FINISHED); } private void doRunUpdateAndExit() throws UpdateException { synchronized (stateLock) { if (state != State.DOWNLOAD_FINISHED) { throw new UpdateException("Must download before run"); } } setState(State.EXECUTING_SETUP); launchExe(); System.exit(0); } private void launchExe() throws UpdateException { try { File exe = new File(TARGET_FILENAME); if (!exe.exists()) { exe = new File(configuration.getTempFolder(), TARGET_FILENAME); } // Use exec(String[]) to avoid space-quoting issues Runtime.getRuntime().exec(new String[] {exe.getAbsolutePath()}); } catch (IOException e) { wrapException("Unable to run update. You may need to manually download it.", e); } } private void assertUpdateIsAvailable() throws UpdateException { synchronized (stateLock) { if (!serverProperties.isStateValid()) { throw new UpdateException("Server error. Try again later."); } if (!isUpdateAvailable()) { throw new UpdateException("Attempt to perform non-existent update"); } } } private void assertNotInErrorState() throws UpdateException { synchronized (stateLock) { if (state == State.ERROR) { throw new UpdateException("Update system must be reset after an error."); } } } private void setState(State value) { synchronized (stateLock) { state = value; if (state == State.DOWNLOAD_FINISHED) { bytesDownloaded = totalBytes; } else if (state != State.DOWNLOAD_IN_PROGRESS) { bytesDownloaded = -1; totalBytes = -1; } if (state != State.ERROR) { errorStateCause = null; } } setChanged(); notifyObservers(); } public boolean isUpdateAvailable() { // TODO (tcox): Make updates work on Linux and Mac return Version.isPmsUpdatable(currentVersion, serverProperties.getLatestVersion()); } private void downloadUpdate() throws UpdateException { String downloadUrl = serverProperties.getDownloadUrl(); /** * Modify the URL to match the Java version. * We keep "Java7" hardcoded to maintain compatibility with * older versions of UMS. */ if (System.getProperty("java.version").startsWith("1.8")) { downloadUrl = downloadUrl.replace("Java7", "Java8"); } else if (System.getProperty("java.version").startsWith("1.6")) { downloadUrl = downloadUrl.replace("Java7", "Java6"); } try { byte[] download = uriRetriever.getWithCallback(downloadUrl, this); writeToDisk(download); } catch (IOException e) { wrapException("Cannot download update", e); } } private void writeToDisk(byte[] download) throws IOException { File target = new File(TARGET_FILENAME); InputStream downloadedFromNetwork = new ByteArrayInputStream(download); FileOutputStream fileOnDisk = null; try { try { fileOnDisk = new FileOutputStream(target); fileOnDisk.write("test".getBytes()); } catch (Exception e) { // seems no rights target = new File(configuration.getTempFolder(), TARGET_FILENAME); } finally { if (fileOnDisk != null) { fileOnDisk.close(); } } fileOnDisk = new FileOutputStream(target); int bytesSaved = IOUtils.copy(downloadedFromNetwork, fileOnDisk); LOGGER.info("Wrote " + bytesSaved + " bytes to " + target.getAbsolutePath()); } finally { IOUtils.closeQuietly(downloadedFromNetwork); IOUtils.closeQuietly(fileOnDisk); } } private void wrapException(String message, Throwable cause) throws UpdateException { throw new UpdateException("Error: " + message, cause); } @Override public void progressMade(String uri, int bytesDownloaded, int totalBytes) throws CancelDownloadException { synchronized (stateLock) { this.bytesDownloaded = bytesDownloaded; this.totalBytes = totalBytes; if (downloadCancelled) { setErrorState(new UpdateException("Download cancelled")); throw new CancelDownloadException(); } } setChanged(); notifyObservers(); } public State getState() { synchronized (stateLock) { return state; } } public Throwable getErrorStateCause() { synchronized (stateLock) { return errorStateCause; } } public int getBytesDownloaded() { synchronized (stateLock) { return bytesDownloaded; } } public int getTotalBytes() { synchronized (stateLock) { return totalBytes; } } public void cancelDownload() { synchronized (stateLock) { downloadCancelled = true; } } public boolean isDownloadCancelled() { synchronized (stateLock) { return downloadCancelled; } } }