/* * This file is part of the Haven & Hearth game client. * Copyright (C) 2009 Fredrik Tolf <fredrik@dolda2000.com>, and * Björn Johannessen <johannessen.bjorn@gmail.com> * * Redistribution and/or modification of this file is subject to the * terms of the GNU Lesser General Public License, version 3, as * published by the Free Software Foundation. * * This program 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 for more details. * * Other parts of this source tree adhere to other copying * rights. Please see the file `COPYING' in the root directory of the * source tree for details. * * A copy the GNU Lesser General Public License is distributed along * with the source tree of which this file is a part in the file * `doc/LPGL-3'. If it is missing for any reason, please see the Free * Software Foundation's website at <http://www.fsf.org/>, or write * to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, * Boston, MA 02111-1307 USA */ package haven; import java.io.ByteArrayOutputStream; import java.io.FileNotFoundException; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.lang.reflect.Method; import java.net.URL; import javax.jnlp.BasicService; import javax.jnlp.FileContents; import javax.jnlp.PersistenceService; import javax.jnlp.ServiceManager; public class JnlpCache implements ResCache { private PersistenceService back; private URL base; private JnlpCache(PersistenceService back, URL base) { this.back = back; this.base = base; } public static JnlpCache create() { try { Class<? extends ServiceManager> cl = Class.forName("javax.jnlp.ServiceManager").asSubclass(ServiceManager.class); Method m = cl.getMethod("lookup", String.class); BasicService basic = (BasicService) m.invoke(null, "javax.jnlp.BasicService"); PersistenceService prs = (PersistenceService) m.invoke(null, "javax.jnlp.PersistenceService"); return (new JnlpCache(prs, basic.getCodeBase())); } catch (Exception e) { return (null); } } private static String mangle(String nm) { StringBuilder buf = new StringBuilder(); for (int i = 0; i < nm.length(); i++) { char c = nm.charAt(i); if (c == '/') buf.append("_"); else buf.append(c); } return (buf.toString()); } private void realput(URL loc, byte[] data) { FileContents file; try { try { file = back.get(loc); } catch (FileNotFoundException e) { back.create(loc, data.length); file = back.get(loc); } if (file.getMaxLength() < data.length) { if (file.setMaxLength(data.length) < data.length) { back.delete(loc); return; } } OutputStream s = file.getOutputStream(true); try { s.write(data); } finally { s.close(); } } catch (IOException e) { return; } catch (Exception e) { /* There seems to be a strange bug in NetX. */ return; } } private void put(final URL loc, final byte[] data) { /* * Interestingly enough, it seems that the JNLP persistence cache, at * least in Sun's implementation, is REALLY SLOW. "Really slow" meaning * that it takes like 1 second to save each muffin. How they managed to * do this, I have no clue, but to counter it, so as to not take up * precious resource/minimap loader time, the actual storage is * deferred. * * Sucks? You don't say. */ Utils.defer(new Runnable() { public void run() { realput(loc, data); } }); } private InputStream get(URL loc) throws IOException { FileContents file = back.get(loc); return (file.getInputStream()); } public OutputStream store(final String name) throws IOException { /* * The persistence service actually yields a real OutputStream, but * since it needs to know the final size of the data before that, it * isn't actually possible to use it as an OutputStream in any * reasonable manner. Thus, all data is first "pre-cached" in a * ByteArrayOutputStream, and then written to the persistence store. * * Oh God, it's so stupid. */ OutputStream ret = new ByteArrayOutputStream() { public void close() { byte[] res = toByteArray(); try { put(new URL(base, mangle(name)), res); } catch (java.net.MalformedURLException e) { throw (new RuntimeException(e)); } } }; return (ret); } public InputStream fetch(String name) throws IOException { try { URL loc = new URL(base, mangle(name)); return (get(loc)); } catch (IOException e) { throw (e); } catch (Exception e) { /* There seems to be a weird bug in NetX */ throw ((IOException) (new IOException("Virtual NetX IO exception").initCause(e))); } } }