package be.nikiroo.fanfix.library; import java.io.IOException; import java.net.URL; import java.util.ArrayList; import java.util.Date; import java.util.List; import be.nikiroo.fanfix.Instance; 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.server.ConnectActionServerObject; import be.nikiroo.utils.serial.server.ServerObject; /** * 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 commands, including PING, will first return a random value to you that * you must hash with your key and return before processing the rest; if the * value not correct, the connection will be closed. *

* BTW: this system is by no means secure. It is just slightly * obfuscated, and operate on clear text (because Google decided not to support * anonymous SSL exchanges on Android, and the main use case for this server is * Android). *

* * @author niki */ public class RemoteLibraryServer extends ServerObject { private final String key; /** * Create a new remote server (will not be active until * {@link RemoteLibraryServer#start()} is called). * * @param key * the key that will restrict access to this server * @param port * the port to listen on * * @throws IOException * in case of I/O error */ public RemoteLibraryServer(String key, int port) throws IOException { super("Fanfix remote library", port, false); this.key = key; setTraceHandler(Instance.getTraceHandler()); } @Override protected Object onRequest(ConnectActionServerObject action, Version clientVersion, Object data) throws Exception { long start = new Date().getTime(); String command = ""; Object[] args = new Object[0]; if (data instanceof Object[]) { Object[] dataArray = (Object[]) data; if (dataArray.length > 0) { command = "" + dataArray[0]; args = new Object[dataArray.length - 1]; for (int i = 1; i < dataArray.length; i++) { args[i - 1] = dataArray[i]; } } } String trace = "[ " + command + "] "; for (Object arg : args) { trace += arg + " "; } System.out.println(trace); // Authentication: String random = StringUtils.getMd5Hash(Double.toString(Math.random())); action.send(random); String answer = "" + action.rec(); if (!answer.equals(RemoteLibrary.hashKey(key, random))) { System.out.println("Key rejected."); action.close(); return null; } // Object rep = doRequest(action, command, args); String rec = StringUtils.formatNumber(action.getBytesReceived()) + "b"; String sent = StringUtils.formatNumber(action.getBytesSent()) + "b"; System.out.println(String.format("[>%s]: (%s sent, %s rec) in %d ms", command, sent, rec, (new Date().getTime() - start))); return rep; } private Object doRequest(ConnectActionServerObject action, String command, Object[] args) throws NoSuchFieldException, NoSuchMethodException, ClassNotFoundException, IOException { if ("PING".equals(command)) { return "PONG"; } else if ("GET_METADATA".equals(command)) { if ("*".equals(args[0])) { Progress pg = createPgForwarder(action); List metas = new ArrayList(); for (MetaData meta : Instance.getLibrary().getMetas(pg)) { MetaData light; if (meta.getCover() == null) { light = meta; } else { light = meta.clone(); light.setCover(null); } metas.add(light); } forcePgDoneSent(pg); return metas.toArray(new MetaData[] {}); } return new MetaData[] { Instance.getLibrary().getInfo( (String) args[0]) }; } else if ("GET_STORY".equals(command)) { MetaData meta = Instance.getLibrary().getInfo((String) args[0]); meta = meta.clone(); meta.setCover(null); action.send(meta); action.rec(); Story story = Instance.getLibrary() .getStory((String) args[0], null); for (Object obj : breakStory(story)) { action.send(obj); action.rec(); } } else if ("SAVE_STORY".equals(command)) { 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.getLibrary().save(story, (String) args[0], null); return story.getMeta().getLuid(); } else if ("IMPORT".equals(command)) { Progress pg = createPgForwarder(action); Story story = Instance.getLibrary().imprt( new URL((String) args[0]), pg); forcePgDoneSent(pg); return story.getMeta().getLuid(); } else if ("DELETE_STORY".equals(command)) { Instance.getLibrary().delete((String) args[0]); } else if ("GET_COVER".equals(command)) { return Instance.getLibrary().getCover((String) args[0]); } else if ("GET_CUSTOM_COVER".equals(command)) { if ("SOURCE".equals(args[0])) { return Instance.getLibrary().getCustomSourceCover( (String) args[1]); } else if ("AUTHOR".equals(args[0])) { return Instance.getLibrary().getCustomAuthorCover( (String) args[1]); } else { return null; } } else if ("SET_COVER".equals(command)) { if ("SOURCE".equals(args[0])) { Instance.getLibrary().setSourceCover((String) args[1], (String) args[2]); } else if ("AUTHOR".equals(args[0])) { Instance.getLibrary().setAuthorCover((String) args[1], (String) args[2]); } } else if ("CHANGE_STA".equals(command)) { Progress pg = createPgForwarder(action); Instance.getLibrary().changeSTA((String) args[0], (String) args[1], (String) args[2], (String) args[3], pg); forcePgDoneSent(pg); } else if ("EXIT".equals(command)) { stop(0, false); } return null; } @Override protected void onError(Exception e) { 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) { if (rep instanceof Integer[]) { Integer[] a = (Integer[]) rep; if (a.length == 3) { int min = a[0]; int max = a[1]; int progress = a[2]; if (min >= 0 && min <= max) { pg.setMinMax(min, max); pg.setProgress(progress); 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 Long[] lastTime = new Long[] { new Date().getTime() }; pg.addProgressListener(new ProgressListener() { @Override public void progress(Progress progress, String name) { int min = pg.getMin(); int max = pg.getMax(); int relativeProgress = min + (int) Math.round(pg.getRelativeProgress() * (max - min)); // Do not re-send the same value twice over the wire, // unless more than 2 seconds have elapsed (to maintain the // connection) if ((p[0] != min || p[1] != max || p[2] != relativeProgress) || (new Date().getTime() - lastTime[0] > 2000)) { p[0] = min; p[1] = max; p[2] = relativeProgress; try { action.send(new Integer[] { min, max, relativeProgress }); action.rec(); } catch (Exception e) { getTraceHandler().error(e); } lastTime[0] = new Date().getTime(); } isDoneForwarded[0] = (pg.getProgress() >= pg.getMax()); } }); return pg; } // 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); } } } }