X-Git-Url: http://git.nikiroo.be/?a=blobdiff_plain;f=src%2Fbe%2Fnikiroo%2Ffanfix%2Flibrary%2FRemoteLibraryServer.java;h=c150a01b6f8a14f68279fc4bb7358f3374a85e10;hb=f433d15308b70e23280a65cef8c54002a7a971ce;hp=566c70fa69a572043ea6749ae5361b05b6ac4b73;hpb=e42573a004fac26378c693ce9ef0d6319713c682;p=nikiroo-utils.git diff --git a/src/be/nikiroo/fanfix/library/RemoteLibraryServer.java b/src/be/nikiroo/fanfix/library/RemoteLibraryServer.java index 566c70f..c150a01 100644 --- a/src/be/nikiroo/fanfix/library/RemoteLibraryServer.java +++ b/src/be/nikiroo/fanfix/library/RemoteLibraryServer.java @@ -1,50 +1,554 @@ package be.nikiroo.fanfix.library; import java.io.IOException; +import java.net.URL; +import java.util.ArrayList; +import java.util.Date; +import java.util.HashMap; import java.util.List; +import java.util.Map; + +import javax.net.ssl.SSLException; import be.nikiroo.fanfix.Instance; +import be.nikiroo.fanfix.bundles.Config; +import be.nikiroo.fanfix.data.Chapter; import be.nikiroo.fanfix.data.MetaData; +import be.nikiroo.fanfix.data.Paragraph; +import be.nikiroo.fanfix.data.Story; +import be.nikiroo.utils.Progress; +import be.nikiroo.utils.Progress.ProgressListener; +import be.nikiroo.utils.StringUtils; import be.nikiroo.utils.Version; -import be.nikiroo.utils.serial.ConnectActionServer; -import be.nikiroo.utils.serial.Server; +import be.nikiroo.utils.serial.server.ConnectActionServerObject; +import be.nikiroo.utils.serial.server.ServerObject; -public class RemoteLibraryServer extends Server { +/** + * Create a new remote server that will listen for orders on the given port. + *

+ * The available commands are given as arrays of objects (first item is the + * command, the rest are the arguments). + *

+ * All the commands are always prefixed by the subkey (which can be EMPTY if + * none). + *

+ *

+ * + * @author niki + */ +public class RemoteLibraryServer extends ServerObject { + private Map commands = new HashMap(); + private Map times = new HashMap(); + private Map wls = new HashMap(); + private Map rws = new HashMap(); - public RemoteLibraryServer(int port) throws IOException { - super(Version.getCurrentVersion(), port, true); + /** + * Create a new remote server (will not be active until + * {@link RemoteLibraryServer#start()} is called). + *

+ * Note: the key we use here is the encryption key (it must not contain a + * subkey). + * + * @throws IOException + * in case of I/O error + */ + public RemoteLibraryServer() throws IOException { + super("Fanfix remote library", + Instance.getInstance().getConfig() + .getInteger(Config.SERVER_PORT), + Instance.getInstance().getConfig() + .getString(Config.SERVER_KEY)); + + setTraceHandler(Instance.getInstance().getTraceHandler()); } @Override - protected Object onRequest(ConnectActionServer action, - Version clientVersion, Object data) throws Exception { - String command = null; - String args = null; - if (data instanceof String) { - command = (String) data; - int pos = command.indexOf(" "); - if (pos >= 0) { - args = command.substring(pos + 1); - command = command.substring(0, pos); + protected Object onRequest(ConnectActionServerObject action, + Version clientVersion, Object data, long id) throws Exception { + long start = new Date().getTime(); + + // defaults are positive (as previous versions without the feature) + boolean rw = true; + boolean wl = true; + + String subkey = ""; + String command = ""; + Object[] args = new Object[0]; + if (data instanceof Object[]) { + Object[] dataArray = (Object[]) data; + if (dataArray.length > 0) { + subkey = "" + dataArray[0]; + } + if (dataArray.length > 1) { + command = "" + dataArray[1]; + + args = new Object[dataArray.length - 2]; + for (int i = 2; i < dataArray.length; i++) { + args[i - 2] = dataArray[i]; + } + } + } + + List whitelist = Instance.getInstance().getConfig().getList(Config.SERVER_WHITELIST); + if (whitelist == null) { + whitelist = new ArrayList(); + } + + if (whitelist.isEmpty()) { + wl = false; + } + + rw = Instance.getInstance().getConfig().getBoolean(Config.SERVER_RW, rw); + if (!subkey.isEmpty()) { + List allowed = Instance.getInstance().getConfig().getList(Config.SERVER_ALLOWED_SUBKEYS); + if (allowed.contains(subkey)) { + if ((subkey + "|").contains("|rw|")) { + rw = true; + } + if ((subkey + "|").contains("|wl|")) { + wl = false; // |wl| = bypass whitelist + whitelist = new ArrayList(); + } } } - System.out.println(String.format("COMMAND: [%s], ARGS: [%s]", command, - args)); + String mode = display(wl, rw); + + String trace = mode + "[ " + command + "] "; + for (Object arg : args) { + trace += arg + " "; + } + long now = System.currentTimeMillis(); + System.out.println(StringUtils.fromTime(now) + ": " + trace); + + Object rep = null; + try { + rep = doRequest(action, command, args, rw, whitelist); + } catch (IOException e) { + rep = new RemoteLibraryException(e, true); + } + + commands.put(id, command); + wls.put(id, wl); + rws.put(id, rw); + times.put(id, (new Date().getTime() - start)); + + return rep; + } + + private String display(boolean whitelist, boolean rw) { + String mode = ""; + if (!rw) { + mode += "RO: "; + } + if (whitelist) { + mode += "WL: "; + } + + return mode; + } + + @Override + protected void onRequestDone(long id, long bytesReceived, long bytesSent) { + boolean whitelist = wls.get(id); + boolean rw = rws.get(id); + wls.remove(id); + rws.remove(id); + + String rec = StringUtils.formatNumber(bytesReceived) + "b"; + String sent = StringUtils.formatNumber(bytesSent) + "b"; + long now = System.currentTimeMillis(); + System.out.println(StringUtils.fromTime(now) + + ": " + + String.format("%s[>%s]: (%s sent, %s rec) in %d ms", + display(whitelist, rw), commands.get(id), sent, rec, + times.get(id))); + + commands.remove(id); + times.remove(id); + } + + private Object doRequest(ConnectActionServerObject action, String command, + Object[] args, boolean rw, List whitelist) + throws NoSuchFieldException, NoSuchMethodException, + ClassNotFoundException, IOException { + if ("PING".equals(command)) { + return rw ? "r/w" : "r/o"; + } else if ("GET_METADATA".equals(command)) { + List metas = new ArrayList(); - if (command != null) { - if (command.equals("GET_METADATA")) { - if (args != null && args.equals("*")) { - List metas = Instance.getLibrary().getMetas(null); - return metas.toArray(new MetaData[] {}); + if ("*".equals(args[0])) { + Progress pg = createPgForwarder(action); + + for (MetaData meta : Instance.getInstance().getLibrary().getMetas(pg)) { + metas.add(removeCover(meta)); + } + + forcePgDoneSent(pg); + } else { + MetaData meta = Instance.getInstance().getLibrary().getInfo((String) args[0]); + MetaData light; + if (meta.getCover() == null) { + light = meta; + } else { + light = meta.clone(); + light.setCover(null); } - } else if (command.equals("GET_STORY")) { - if (args != null) { - return Instance.getLibrary().getStory(args, null); + + metas.add(light); + } + + if (!whitelist.isEmpty()) { + for (int i = 0; i < metas.size(); i++) { + if (!whitelist.contains(metas.get(i).getSource())) { + metas.remove(i); + i--; + } + } + } + + return metas.toArray(new MetaData[0]); + } else if ("GET_STORY".equals(command)) { + MetaData meta = Instance.getInstance().getLibrary().getInfo((String) args[0]); + if (meta == null) { + return null; + } + + if (!whitelist.isEmpty()) { + if (!whitelist.contains(meta.getSource())) { + return null; } } + + meta = meta.clone(); + meta.setCover(null); + + action.send(meta); + action.rec(); + + Story story = Instance.getInstance().getLibrary().getStory((String) args[0], null); + for (Object obj : breakStory(story)) { + action.send(obj); + action.rec(); + } + } else if ("SAVE_STORY".equals(command)) { + if (!rw) { + throw new RemoteLibraryException("Read-Only remote library: " + + args[0], false); + } + + List list = new ArrayList(); + + action.send(null); + Object obj = action.rec(); + while (obj != null) { + list.add(obj); + action.send(null); + obj = action.rec(); + } + + Story story = rebuildStory(list); + Instance.getInstance().getLibrary().save(story, (String) args[0], null); + return story.getMeta().getLuid(); + } else if ("IMPORT".equals(command)) { + if (!rw) { + throw new RemoteLibraryException("Read-Only remote library: " + + args[0], false); + } + + Progress pg = createPgForwarder(action); + MetaData meta = Instance.getInstance().getLibrary().imprt(new URL((String) args[0]), pg); + forcePgDoneSent(pg); + return meta.getLuid(); + } else if ("DELETE_STORY".equals(command)) { + if (!rw) { + throw new RemoteLibraryException("Read-Only remote library: " + + args[0], false); + } + + Instance.getInstance().getLibrary().delete((String) args[0]); + } else if ("GET_COVER".equals(command)) { + return Instance.getInstance().getLibrary().getCover((String) args[0]); + } else if ("GET_CUSTOM_COVER".equals(command)) { + if ("SOURCE".equals(args[0])) { + return Instance.getInstance().getLibrary().getCustomSourceCover((String) args[1]); + } else if ("AUTHOR".equals(args[0])) { + return Instance.getInstance().getLibrary().getCustomAuthorCover((String) args[1]); + } else { + return null; + } + } else if ("SET_COVER".equals(command)) { + if (!rw) { + throw new RemoteLibraryException("Read-Only remote library: " + + args[0] + ", " + args[1], false); + } + + if ("SOURCE".equals(args[0])) { + Instance.getInstance().getLibrary().setSourceCover((String) args[1], (String) args[2]); + } else if ("AUTHOR".equals(args[0])) { + Instance.getInstance().getLibrary().setAuthorCover((String) args[1], (String) args[2]); + } + } else if ("CHANGE_STA".equals(command)) { + if (!rw) { + throw new RemoteLibraryException("Read-Only remote library: " + args[0] + ", " + args[1], false); + } + + Progress pg = createPgForwarder(action); + Instance.getInstance().getLibrary().changeSTA((String) args[0], (String) args[1], (String) args[2], + (String) args[3], pg); + forcePgDoneSent(pg); + } else if ("EXIT".equals(command)) { + if (!rw) { + throw new RemoteLibraryException( + "Read-Only remote library: EXIT", false); + } + + stop(10000, false); } return null; } + + @Override + protected void onError(Exception e) { + if (e instanceof SSLException) { + long now = System.currentTimeMillis(); + System.out.println(StringUtils.fromTime(now) + ": " + + "[Client connection refused (bad key)]"); + } else { + getTraceHandler().error(e); + } + } + + /** + * Break a story in multiple {@link Object}s for easier serialisation. + * + * @param story + * the {@link Story} to break + * + * @return the list of {@link Object}s + */ + static List breakStory(Story story) { + List list = new ArrayList(); + + story = story.clone(); + list.add(story); + + if (story.getMeta().isImageDocument()) { + for (Chapter chap : story) { + list.add(chap); + list.addAll(chap.getParagraphs()); + chap.setParagraphs(new ArrayList()); + } + story.setChapters(new ArrayList()); + } + + return list; + } + + /** + * Rebuild a story from a list of broke up {@link Story} parts. + * + * @param list + * the list of {@link Story} parts + * + * @return the reconstructed {@link Story} + */ + static Story rebuildStory(List list) { + Story story = null; + Chapter chap = null; + + for (Object obj : list) { + if (obj instanceof Story) { + story = (Story) obj; + } else if (obj instanceof Chapter) { + chap = (Chapter) obj; + story.getChapters().add(chap); + } else if (obj instanceof Paragraph) { + chap.getParagraphs().add((Paragraph) obj); + } + } + + return story; + } + + /** + * Update the {@link Progress} with the adequate {@link Object} received + * from the network via {@link RemoteLibraryServer}. + * + * @param pg + * the {@link Progress} to update + * @param rep + * the object received from the network + * + * @return TRUE if it was a progress event, FALSE if not + */ + static boolean updateProgress(Progress pg, Object rep) { + boolean updateProgress = false; + if (rep instanceof Integer[] && ((Integer[]) rep).length == 3) + updateProgress = true; + if (rep instanceof Object[] && ((Object[]) rep).length >= 5 + && "UPDATE".equals(((Object[]) rep)[0])) + updateProgress = true; + + if (updateProgress) { + Object[] a = (Object[]) rep; + + int offset = 0; + if (a[0] instanceof String) { + offset = 1; + } + + int min = (Integer) a[0 + offset]; + int max = (Integer) a[1 + offset]; + int progress = (Integer) a[2 + offset]; + + Object meta = null; + if (a.length > (3 + offset)) { + meta = a[3 + offset]; + } + + String name = null; + if (a.length > (4 + offset)) { + name = a[4 + offset] == null ? "" : a[4 + offset].toString(); + } + + + if (min >= 0 && min <= max) { + pg.setName(name); + pg.setMinMax(min, max); + pg.setProgress(progress); + if (meta != null) { + pg.put("meta", meta); + } + + return true; + } + } + + return false; + } + + /** + * Create a {@link Progress} that will forward its progress over the + * network. + * + * @param action + * the {@link ConnectActionServerObject} to use to forward it + * + * @return the {@link Progress} + */ + private Progress createPgForwarder(final ConnectActionServerObject action) { + final Boolean[] isDoneForwarded = new Boolean[] { false }; + final Progress pg = new Progress() { + @Override + public boolean isDone() { + return isDoneForwarded[0]; + } + }; + + final Integer[] p = new Integer[] { -1, -1, -1 }; + final Object[] pMeta = new MetaData[1]; + final String[] pName = new String[1]; + final Long[] lastTime = new Long[] { new Date().getTime() }; + pg.addProgressListener(new ProgressListener() { + @Override + public void progress(Progress progress, String name) { + Object meta = pg.get("meta"); + if (meta instanceof MetaData) { + meta = removeCover((MetaData)meta); + } + + int min = pg.getMin(); + int max = pg.getMax(); + int rel = min + + (int) Math.round(pg.getRelativeProgress() + * (max - min)); + + boolean samePg = p[0] == min && p[1] == max && p[2] == rel; + + // Do not re-send the same value twice over the wire, + // unless more than 2 seconds have elapsed (to maintain the + // connection) + if (!samePg || !same(pMeta[0], meta) + || !same(pName[0], name) // + || (new Date().getTime() - lastTime[0] > 2000)) { + p[0] = min; + p[1] = max; + p[2] = rel; + pMeta[0] = meta; + pName[0] = name; + + try { + action.send(new Object[] { "UPDATE", min, max, rel, + meta, name }); + action.rec(); + } catch (Exception e) { + getTraceHandler().error(e); + } + + lastTime[0] = new Date().getTime(); + } + + isDoneForwarded[0] = (pg.getProgress() >= pg.getMax()); + } + }); + + return pg; + } + + private boolean same(Object obj1, Object obj2) { + if (obj1 == null || obj2 == null) + return obj1 == null && obj2 == null; + + return obj1.equals(obj2); + } + + // with 30 seconds timeout + private void forcePgDoneSent(Progress pg) { + long start = new Date().getTime(); + pg.done(); + while (!pg.isDone() && new Date().getTime() - start < 30000) { + try { + Thread.sleep(100); + } catch (InterruptedException e) { + getTraceHandler().error(e); + } + } + } + + private MetaData removeCover(MetaData meta) { + MetaData light = null; + if (meta != null) { + if (meta.getCover() == null) { + light = meta; + } else { + light = meta.clone(); + light.setCover(null); + } + } + + return light; + } }