/* * Copyright 2012 Red Hat, Inc. * This file is part of IcedTea, http://icedtea.classpath.org * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Sun designates this * particular file as subject to the "Classpath" exception as provided * by Sun in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. */ package net.sourceforge.jnlp; import java.io.ByteArrayInputStream; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.net.MalformedURLException; import java.net.URL; import java.util.ArrayList; import java.util.Arrays; import java.util.HashSet; import java.util.List; import java.util.Locale; import java.util.Map; import java.util.Set; import net.sourceforge.jnlp.runtime.JNLPRuntime; import net.sourceforge.jnlp.util.logging.OutputController; import net.sourceforge.jnlp.util.replacements.BASE64Decoder; /** * Allows reuse of code that expects a JNLPFile object, * while overriding behaviour specific to applets. */ public class PluginBridge extends JNLPFile { private PluginParameters params; private Set jars = new HashSet(); private List extensionJars = new ArrayList(); //Folders can be added to the code-base through the archive tag private List codeBaseFolders = new ArrayList(); private String[] cacheJars = new String[0]; private String[] cacheExJars = new String[0]; private boolean usePack; private boolean useVersion; private boolean useJNLPHref; /** * Creates a new PluginBridge using a default JNLPCreator. */ public PluginBridge(URL codebase, URL documentBase, String jar, String main, int width, int height, PluginParameters params) throws Exception { this(codebase, documentBase, jar, main, width, height, params, new JNLPCreator()); } /** * Handles archive tag entries, which may be folders or jar files * @param archives the components of the archive tag */ private void addArchiveEntries(String[] archives) { for (String archiveEntry : archives){ // trim white spaces archiveEntry = archiveEntry.trim(); /*Only '/' on linux, '/' or '\\' on windows*/ if (archiveEntry.endsWith("/") || archiveEntry.endsWith(File.pathSeparator)) { this.codeBaseFolders.add(archiveEntry); } else { this.jars.add(archiveEntry); } } } public PluginBridge(URL codebase, URL documentBase, String archive, String main, int width, int height, PluginParameters params, JNLPCreator jnlpCreator) throws Exception { specVersion = new Version("1.0"); fileVersion = new Version("1.1"); this.codeBase = codebase; this.sourceLocation = documentBase; this.params = params; this.parserSettings = ParserSettings.getGlobalParserSettings(); if (params.getJNLPHref() != null) { useJNLPHref = true; try { // Use codeBase as the context for the URL. If jnlp_href's // value is a complete URL, it will replace codeBase's context. ParserSettings defaultSettings = new ParserSettings(); URL jnlp = new URL(codeBase, params.getJNLPHref()); JNLPFile jnlpFile = null; if (params.getJNLPEmbedded() != null) { InputStream jnlpInputStream = new ByteArrayInputStream(decodeBase64String(params.getJNLPEmbedded())); jnlpFile = new JNLPFile(jnlpInputStream, codeBase, defaultSettings); } else { jnlpFile = jnlpCreator.create(jnlp, null, defaultSettings, JNLPRuntime.getDefaultUpdatePolicy(), codeBase); } if (jnlpFile.isApplet()) main = jnlpFile.getApplet().getMainClass(); Map jnlpParams = jnlpFile.getApplet().getParameters(); info = jnlpFile.info; // Change the parameter name to lowercase to follow conventions. for (Map.Entry entry : jnlpParams.entrySet()) { this.params.put(entry.getKey().toLowerCase(), entry.getValue()); } JARDesc[] jarDescs = jnlpFile.getResources().getJARs(); for (JARDesc jarDesc : jarDescs) { String fileName = jarDesc.getLocation().toExternalForm(); this.jars.add(fileName); } // Store any extensions listed in the JNLP file to be returned later on, namely in getResources() extensionJars = Arrays.asList(jnlpFile.getResources().getExtensions()); } catch (MalformedURLException e) { // Don't fail because we cannot get the jnlp file. Parameters are optional not required. // it is the site developer who should ensure that file exist. OutputController.getLogger().log(OutputController.Level.ERROR_ALL, "Unable to get JNLP file at: " + params.getJNLPHref() + " with context of URL as: " + codeBase.toExternalForm()); } } else { // Should we populate this list with applet attribute tags? info = new ArrayList(); useJNLPHref = false; } // also, see if cache_archive is specified String cacheArchive = params.getCacheArchive(); if (!cacheArchive.isEmpty()) { String[] versions = new String[0]; // are there accompanying versions? String cacheVersion = params.getCacheVersion(); if (!cacheVersion.isEmpty()) { versions = cacheVersion.split(","); } String[] jars = cacheArchive.split(","); cacheJars = new String[jars.length]; for (int i = 0; i < jars.length; i++) { cacheJars[i] = jars[i].trim(); if (versions.length > 0) { cacheJars[i] += ";" + versions[i].trim(); } } } String cacheArchiveEx = params.getCacheArchiveEx(); if (!cacheArchiveEx.isEmpty()) { cacheExJars = cacheArchiveEx.split(","); } if (archive != null && archive.length() > 0) { String[] archives = archive.split(","); addArchiveEntries(archives); OutputController.getLogger().log(OutputController.Level.ERROR_DEBUG, "Jar string: " + archive); OutputController.getLogger().log(OutputController.Level.ERROR_DEBUG, "jars length: " + archives.length); } if (main.endsWith(".class")) main = main.substring(0, main.length() - 6); // the class name should be of the form foo.bar.Baz not foo/bar/Baz String mainClass = main.replace('/', '.'); launchType = new AppletDesc(getTitle(), mainClass, documentBase, width, height, params.getUnmodifiableMap()); if (main.endsWith(".class")) //single class file only security = new SecurityDesc(this, SecurityDesc.SANDBOX_PERMISSIONS, codebase.getHost()); else security = null; this.uniqueKey = params.getUniqueKey(codebase); usePack = false; useVersion = false; String jargs = params.getJavaArguments(); if (!jargs.isEmpty()) { for (String s : jargs.split(" ")) { String[] parts = s.trim().split("="); if (parts.length == 2 && Boolean.valueOf(parts[1])) { if ("-Djnlp.packEnabled".equals(parts[0])) { usePack = true; } else if ("-Djnlp.versionEnabled".equals(parts[0])) { useVersion = true; } } } } } public List getArchiveJars() { return new ArrayList(jars); } public boolean codeBaseLookup() { return params.useCodebaseLookup(); } public boolean useJNLPHref() { return useJNLPHref; } /** * {@inheritDoc } */ @Override public DownloadOptions getDownloadOptions() { return new DownloadOptions(usePack, useVersion); } @Override public String getTitle() { String inManifestTitle = super.getTitleFromManifest(); if (inManifestTitle != null) { return inManifestTitle; } //specification is recommending main class instead of html parameter //http://docs.oracle.com/javase/7/docs/technotes/guides/jweb/manifest.html#app_name String mainClass = getManifestsAttributes().getMainClass(); if (mainClass != null) { return mainClass; } return params.getAppletTitle(); } public ResourcesDesc getResources(final Locale locale, final String os, final String arch) { return new ResourcesDesc(this, new Locale[] { locale }, new String[] { os }, new String[] { arch }) { @Override public List getResources(Class launchType) { // Need to add the JAR manually... //should this be done to sharedResources on init? if (launchType.equals(JARDesc.class)) { try { List jarDescs = new ArrayList(); jarDescs.addAll(sharedResources.getResources(JARDesc.class)); for (String name : jars) { if (name.length() > 0) jarDescs.add(new JARDesc(new URL(codeBase, name), null, null, false, true, false, true)); } boolean cacheable = true; if (params.getCacheOption().equalsIgnoreCase("no")) cacheable = false; for (String cacheJar : cacheJars) { String[] jarAndVer = cacheJar.split(";"); String jar = jarAndVer[0]; Version version = null; if (jar.length() == 0) continue; if (jarAndVer.length > 1) { version = new Version(jarAndVer[1]); } jarDescs.add(new JARDesc(new URL(codeBase, jar), version, null, false, true, false, cacheable)); } for (String cacheExJar : cacheExJars) { if (cacheExJar.length() == 0) continue; String[] jarInfo = cacheExJar.split(";"); String jar = jarInfo[0].trim(); Version version = null; boolean lazy = true; if (jarInfo.length > 1) { // format is name[[;preload];version] if (jarInfo[1].equals("preload")) { lazy = false; } else { version = new Version(jarInfo[1].trim()); } if (jarInfo.length > 2) { lazy = false; version = new Version(jarInfo[2].trim()); } } jarDescs.add(new JARDesc(new URL(codeBase, jar), version, null, lazy, true, false, false)); } // We know this is a safe list of JarDesc objects @SuppressWarnings("unchecked") List result = (List) jarDescs; return result; } catch (MalformedURLException ex) { /* Ignored */ } } else if (launchType.equals(ExtensionDesc.class)) { // We hope this is a safe list of JarDesc objects @SuppressWarnings("unchecked") List castList = (List) extensionJars; // this list is populated when the PluginBridge is first constructed return castList; } return sharedResources.getResources(launchType); } @Override public JARDesc[] getJARs() { List jarDescs = getResources(JARDesc.class); return jarDescs.toArray(new JARDesc[jarDescs.size()]); } public void addResource(Object resource) { // todo: honor the current locale, os, arch values sharedResources.addResource(resource); } }; } /** * Returns the list of folders to be added to the codebase */ public List getCodeBaseFolders() { return new ArrayList(codeBaseFolders); } /** * Returns the resources section of the JNLP file for the * specified locale, os, and arch. */ public ResourcesDesc[] getResourcesDescs(final Locale locale, final String os, final String arch) { return new ResourcesDesc[] { getResources(locale, os, arch) }; } public boolean isApplet() { return true; } public boolean isApplication() { return false; } public boolean isComponent() { return false; } public boolean isInstaller() { return false; } /** * Returns the decoded BASE64 string */ static byte[] decodeBase64String(String encodedString) throws IOException { BASE64Decoder base64 = new BASE64Decoder(); return base64.decodeBuffer(encodedString); } }