org.transdroid.daemon.Synology.SynologyAdapter.java Source code

Java tutorial

Introduction

Here is the source code for org.transdroid.daemon.Synology.SynologyAdapter.java

Source

/*
 *   This file is part of Transdroid <http://www.transdroid.org>
 *   
 *   Transdroid is free software: you can redistribute it and/or modify
 *   it under the terms of the GNU General Public License as published by
 *   the Free Software Foundation, either version 3 of the License, or
 *   (at your option) any later version.
 *   
 *   Transdroid 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.
 *   
 *   You should have received a copy of the GNU General Public License
 *   along with Transdroid.  If not, see <http://www.gnu.org/licenses/>.
 *   
 */
package org.transdroid.daemon.Synology;

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import org.transdroid.core.gui.log.Log;
import org.transdroid.daemon.Daemon;
import org.transdroid.daemon.DaemonException;
import org.transdroid.daemon.DaemonException.ExceptionType;
import org.transdroid.daemon.DaemonSettings;
import org.transdroid.daemon.IDaemonAdapter;
import org.transdroid.daemon.Priority;
import org.transdroid.daemon.Torrent;
import org.transdroid.daemon.TorrentDetails;
import org.transdroid.daemon.TorrentFile;
import org.transdroid.daemon.TorrentStatus;
import org.transdroid.daemon.task.AddByMagnetUrlTask;
import org.transdroid.daemon.task.AddByUrlTask;
import org.transdroid.daemon.task.DaemonTask;
import org.transdroid.daemon.task.DaemonTaskFailureResult;
import org.transdroid.daemon.task.DaemonTaskResult;
import org.transdroid.daemon.task.DaemonTaskSuccessResult;
import org.transdroid.daemon.task.GetFileListTask;
import org.transdroid.daemon.task.GetFileListTaskSuccessResult;
import org.transdroid.daemon.task.GetTorrentDetailsTask;
import org.transdroid.daemon.task.GetTorrentDetailsTaskSuccessResult;
import org.transdroid.daemon.task.RetrieveTask;
import org.transdroid.daemon.task.RetrieveTaskSuccessResult;
import org.transdroid.daemon.task.SetTransferRatesTask;
import org.transdroid.daemon.util.Collections2;
import org.transdroid.daemon.util.HttpHelper;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;

/**
 * The daemon adapter from the Synology Download Station torrent client.
 */
public class SynologyAdapter implements IDaemonAdapter {

    private static final String LOG_NAME = "Synology daemon";

    private DaemonSettings settings;
    private DefaultHttpClient httpClient;

    private String sid;

    public SynologyAdapter(DaemonSettings settings) {
        this.settings = settings;
    }

    @Override
    public DaemonTaskResult executeTask(Log log, DaemonTask task) {
        String tid;
        try {
            switch (task.getMethod()) {
            case Retrieve:
                return new RetrieveTaskSuccessResult((RetrieveTask) task, tasksList(log), null);
            case GetStats:
                return null;
            case GetTorrentDetails:
                tid = task.getTargetTorrent().getUniqueID();
                return new GetTorrentDetailsTaskSuccessResult((GetTorrentDetailsTask) task,
                        torrentDetails(log, tid));
            case GetFileList:
                tid = task.getTargetTorrent().getUniqueID();
                return new GetFileListTaskSuccessResult((GetFileListTask) task, fileList(log, tid));
            case AddByFile:
                return null;
            case AddByUrl:
                String url = ((AddByUrlTask) task).getUrl();
                createTask(log, url);
                return new DaemonTaskSuccessResult(task);
            case AddByMagnetUrl:
                String magnet = ((AddByMagnetUrlTask) task).getUrl();
                createTask(log, magnet);
                return new DaemonTaskSuccessResult(task);
            case Remove:
                tid = task.getTargetTorrent().getUniqueID();
                removeTask(log, tid);
                return new DaemonTaskSuccessResult(task);
            case Pause:
                tid = task.getTargetTorrent().getUniqueID();
                pauseTask(log, tid);
                return new DaemonTaskSuccessResult(task);
            case PauseAll:
                pauseAllTasks(log);
                return new DaemonTaskSuccessResult(task);
            case Resume:
                tid = task.getTargetTorrent().getUniqueID();
                resumeTask(log, tid);
                return new DaemonTaskSuccessResult(task);
            case ResumeAll:
                resumeAllTasks(log);
                return new DaemonTaskSuccessResult(task);
            case SetDownloadLocation:
                return null;
            case SetFilePriorities:
                return null;
            case SetTransferRates:
                SetTransferRatesTask ratesTask = (SetTransferRatesTask) task;
                int uploadRate = ratesTask.getUploadRate() == null ? 0 : ratesTask.getUploadRate();
                int downloadRate = ratesTask.getDownloadRate() == null ? 0 : ratesTask.getDownloadRate();
                setTransferRates(log, uploadRate, downloadRate);
                return new DaemonTaskSuccessResult(task);
            case SetAlternativeMode:
            default:
                return null;
            }
        } catch (DaemonException e) {
            return new DaemonTaskFailureResult(task, e);
        }
    }

    @Override
    public Daemon getType() {
        return settings.getType();
    }

    @Override
    public DaemonSettings getSettings() {
        return this.settings;
    }

    // Synology API

    private String login(Log log) throws DaemonException {
        log.d(LOG_NAME, "login()");
        try {
            return new SynoRequest("auth.cgi", "SYNO.API.Auth", "2")
                    .get("&method=login&account=" + settings.getUsername() + "&passwd=" + settings.getPassword()
                            + "&session=DownloadStation&format=sid")
                    .getData(log).getString("sid");
        } catch (JSONException e) {
            throw new DaemonException(ExceptionType.ParsingFailed, e.toString());
        }
    }

    private void setTransferRates(Log log, int uploadRate, int downloadRate) throws DaemonException {
        authGet(log, "SYNO.DownloadStation.Info", "1", "DownloadStation/info.cgi",
                "&method=setserverconfig&bt_max_upload=" + uploadRate + "&bt_max_download=" + downloadRate)
                        .ensureSuccess(log);
    }

    private void createTask(Log log, String uri) throws DaemonException {
        try {
            authGet(log, "SYNO.DownloadStation.Task", "1", "DownloadStation/task.cgi",
                    "&method=create&uri=" + URLEncoder.encode(uri, "UTF-8")).ensureSuccess(log);
        } catch (UnsupportedEncodingException e) {
            // Never happens
            throw new DaemonException(ExceptionType.UnexpectedResponse, e.toString());
        }
    }

    private void removeTask(Log log, String tid) throws DaemonException {
        List<String> tids = new ArrayList<String>();
        tids.add(tid);
        removeTasks(log, tids);
    }

    private void pauseTask(Log log, String tid) throws DaemonException {
        List<String> tids = new ArrayList<String>();
        tids.add(tid);
        pauseTasks(log, tids);
    }

    private void resumeTask(Log log, String tid) throws DaemonException {
        List<String> tids = new ArrayList<String>();
        tids.add(tid);
        resumeTasks(log, tids);
    }

    private void pauseAllTasks(Log log) throws DaemonException {
        List<String> tids = new ArrayList<String>();
        for (Torrent torrent : tasksList(log)) {
            tids.add(torrent.getUniqueID());
        }
        pauseTasks(log, tids);
    }

    private void resumeAllTasks(Log log) throws DaemonException {
        List<String> tids = new ArrayList<String>();
        for (Torrent torrent : tasksList(log)) {
            tids.add(torrent.getUniqueID());
        }
        resumeTasks(log, tids);
    }

    private void removeTasks(Log log, List<String> tids) throws DaemonException {
        authGet(log, "SYNO.DownloadStation.Task", "1", "DownloadStation/task.cgi",
                "&method=delete&id=" + Collections2.joinString(tids, ",") + "").ensureSuccess(log);
    }

    private void pauseTasks(Log log, List<String> tids) throws DaemonException {
        authGet(log, "SYNO.DownloadStation.Task", "1", "DownloadStation/task.cgi",
                "&method=pause&id=" + Collections2.joinString(tids, ",")).ensureSuccess(log);
    }

    private void resumeTasks(Log log, List<String> tids) throws DaemonException {
        authGet(log, "SYNO.DownloadStation.Task", "1", "DownloadStation/task.cgi",
                "&method=resume&id=" + Collections2.joinString(tids, ",")).ensureSuccess(log);
    }

    private List<Torrent> tasksList(Log log) throws DaemonException {
        try {
            JSONArray jsonTasks = authGet(log, "SYNO.DownloadStation.Task", "1", "DownloadStation/task.cgi",
                    "&method=list&additional=detail,transfer,tracker").getData(log).getJSONArray("tasks");
            log.d(LOG_NAME, "Tasks = " + jsonTasks.toString());
            List<Torrent> result = new ArrayList<Torrent>();
            for (int i = 0; i < jsonTasks.length(); i++) {
                result.add(parseTorrent(i, jsonTasks.getJSONObject(i)));
            }
            return result;
        } catch (JSONException e) {
            throw new DaemonException(ExceptionType.ParsingFailed, e.toString());
        }
    }

    private List<TorrentFile> fileList(Log log, String torrentId) throws DaemonException {
        try {
            List<TorrentFile> result = new ArrayList<TorrentFile>();
            JSONObject jsonTask = authGet(log, "SYNO.DownloadStation.Task", "1", "DownloadStation/task.cgi",
                    "&method=getinfo&id=" + torrentId + "&additional=detail,transfer,tracker,file").getData(log)
                            .getJSONArray("tasks").getJSONObject(0);
            log.d(LOG_NAME, "File list = " + jsonTask.toString());
            JSONObject additional = jsonTask.getJSONObject("additional");
            if (!additional.has("file")) {
                return result;
            }
            JSONArray files = additional.getJSONArray("file");
            for (int i = 0; i < files.length(); i++) {
                JSONObject task = files.getJSONObject(i);
                // @formatter:off
                result.add(new TorrentFile(task.getString("filename"), task.getString("filename"), null, null,
                        task.getLong("size"), task.getLong("size_downloaded"), priority(task.getString("priority"))
                // @formatter:on
                ));
            }
            return result;
        } catch (JSONException e) {
            throw new DaemonException(ExceptionType.ParsingFailed, e.toString());
        }
    }

    private TorrentDetails torrentDetails(Log log, String torrentId) throws DaemonException {
        List<String> trackers = new ArrayList<String>();
        List<String> errors = new ArrayList<String>();
        try {
            JSONObject jsonTorrent = authGet(log, "SYNO.DownloadStation.Task", "1", "DownloadStation/task.cgi",
                    "&method=getinfo&id=" + torrentId + "&additional=tracker").getData(log).getJSONArray("tasks")
                            .getJSONObject(0);
            JSONObject additional = jsonTorrent.getJSONObject("additional");
            if (additional.has("tracker")) {
                JSONArray tracker = additional.getJSONArray("tracker");
                for (int i = 0; i < tracker.length(); i++) {
                    JSONObject t = tracker.getJSONObject(i);
                    if ("Success".equals(t.getString("status"))) {
                        trackers.add(t.getString("url"));
                    } else {
                        errors.add(t.getString("status"));
                    }
                }
            }
            return new TorrentDetails(trackers, errors);
        } catch (JSONException e) {
            throw new DaemonException(ExceptionType.ParsingFailed, e.toString());
        }
    }

    private Torrent parseTorrent(long id, JSONObject jsonTorrent) throws JSONException, DaemonException {
        JSONObject additional = jsonTorrent.getJSONObject("additional");
        JSONObject detail = additional.getJSONObject("detail");
        JSONObject transfer = additional.getJSONObject("transfer");
        long downloaded = transfer.getLong("size_downloaded");
        int speed = transfer.getInt("speed_download");
        long size = jsonTorrent.getLong("size");
        Float eta = Float.valueOf(size - downloaded) / speed;
        int totalSeeders = 0;
        int totalLeechers = 0;
        if (additional.has("tracker")) {
            JSONArray tracker = additional.getJSONArray("tracker");
            for (int i = 0; i < tracker.length(); i++) {
                JSONObject t = tracker.getJSONObject(i);
                if ("Success".equals(t.getString("status"))) {
                    totalLeechers += t.getInt("peers");
                    totalSeeders += t.getInt("seeds");
                }
            }
        }
        // @formatter:off
        return new Torrent(id, jsonTorrent.getString("id"), jsonTorrent.getString("title"),
                torrentStatus(jsonTorrent.getString("status")), detail.getString("destination"), speed,
                transfer.getInt("speed_upload"), detail.getInt("connected_seeders"), totalSeeders,
                detail.getInt("connected_leechers"), totalLeechers, eta.intValue(), downloaded,
                transfer.getLong("size_uploaded"), size, (size == 0) ? 0 : (Float.valueOf(downloaded) / size), 0,
                jsonTorrent.getString("title"), new Date(detail.getLong("create_time") * 1000), null, "",
                settings.getType()
        // @formatter:on
        );
    }

    private TorrentStatus torrentStatus(String status) {
        if ("downloading".equals(status)) {
            return TorrentStatus.Downloading;
        }
        if ("seeding".equals(status)) {
            return TorrentStatus.Seeding;
        }
        if ("finished".equals(status)) {
            return TorrentStatus.Paused;
        }
        if ("finishing".equals(status)) {
            return TorrentStatus.Paused;
        }
        if ("waiting".equals(status)) {
            return TorrentStatus.Waiting;
        }
        if ("paused".equals(status)) {
            return TorrentStatus.Paused;
        }
        if ("error".equals(status)) {
            return TorrentStatus.Error;
        }
        return TorrentStatus.Unknown;
    }

    private Priority priority(String priority) {
        if ("low".equals(priority)) {
            return Priority.Low;
        }
        if ("normal".equals(priority)) {
            return Priority.Normal;
        }
        if ("high".equals(priority)) {
            return Priority.High;
        }
        return Priority.Off;
    }

    /**
     * Authenticated GET. If no session open, a login authGet will be done before-hand.
     */
    private SynoResponse authGet(Log log, String api, String version, String path, String params)
            throws DaemonException {
        if (sid == null) {
            sid = login(log);
        }
        return new SynoRequest(path, api, version).get(params + "&_sid=" + sid);
    }

    private DefaultHttpClient getHttpClient() throws DaemonException {
        if (httpClient == null) {
            httpClient = HttpHelper.createStandardHttpClient(settings, true);
        }
        return httpClient;
    }

    private static class SynoResponse {

        private final HttpResponse response;

        public SynoResponse(HttpResponse response) {
            this.response = response;
        }

        public JSONObject getData(Log log) throws DaemonException {
            JSONObject json = getJson(log);
            try {
                if (json.getBoolean("success")) {
                    return json.getJSONObject("data");
                } else {
                    log.e(LOG_NAME, "not a success: " + json.toString());
                    throw new DaemonException(ExceptionType.AuthenticationFailure, json.getString("error"));
                }
            } catch (JSONException e) {
                throw new DaemonException(ExceptionType.ParsingFailed, e.toString());
            }
        }

        public JSONObject getJson(Log log) throws DaemonException {
            try {
                HttpEntity entity = response.getEntity();
                if (entity == null) {
                    log.e(LOG_NAME, "Error: No entity in HTTP response");
                    throw new DaemonException(ExceptionType.UnexpectedResponse,
                            "No HTTP entity object in response.");
                }
                // Read JSON response
                java.io.InputStream instream = entity.getContent();
                String result = HttpHelper.convertStreamToString(instream);
                JSONObject json;
                json = new JSONObject(result);
                instream.close();
                return json;
            } catch (JSONException e) {
                throw new DaemonException(ExceptionType.UnexpectedResponse, "Bad JSON");
            } catch (IOException e) {
                log.e(LOG_NAME, "getJson error: " + e.toString());
                throw new DaemonException(ExceptionType.AuthenticationFailure, e.toString());
            }
        }

        public void ensureSuccess(Log log) throws DaemonException {
            JSONObject json = getJson(log);
            try {
                if (!json.getBoolean("success")) {
                    throw new DaemonException(ExceptionType.UnexpectedResponse, json.getString("error"));
                }
            } catch (JSONException e) {
                throw new DaemonException(ExceptionType.ParsingFailed, e.toString());
            }
        }

    }

    private class SynoRequest {
        private final String path;
        private final String api;
        private final String version;

        public SynoRequest(String path, String api, String version) {
            this.path = path;
            this.api = api;
            this.version = version;
        }

        public SynoResponse get(String params) throws DaemonException {
            try {
                return new SynoResponse(getHttpClient().execute(new HttpGet(buildURL(params))));
            } catch (IOException e) {
                throw new DaemonException(ExceptionType.ConnectionError, e.toString());
            }
        }

        private String buildURL(String params) {
            return (settings.getSsl() ? "https://" : "http://") + settings.getAddress() + ":" + settings.getPort()
                    + "/webapi/" + path + "?api=" + api + "&version=" + version + params;
        }

    }

}