mirror of
https://github.com/aria2/aria2.git
synced 2025-04-04 21:17:41 +03:00
Added JSON-RPC over WebSocket.
Wslay library must be placed under deps/wslay.
This commit is contained in:
parent
f16511012e
commit
f4e2c7f060
33 changed files with 1476 additions and 227 deletions
|
@ -1,4 +1,4 @@
|
||||||
SUBDIRS = po intl lib src test doc
|
SUBDIRS = po intl lib deps src test doc
|
||||||
|
|
||||||
ACLOCAL_AMFLAGS = -I m4 --install
|
ACLOCAL_AMFLAGS = -I m4 --install
|
||||||
|
|
||||||
|
|
10
configure.ac
10
configure.ac
|
@ -2,7 +2,9 @@
|
||||||
# Process this file with autoconf to produce a configure script.
|
# Process this file with autoconf to produce a configure script.
|
||||||
#
|
#
|
||||||
AC_PREREQ([2.67])
|
AC_PREREQ([2.67])
|
||||||
|
LT_PREREQ([2.2.6])
|
||||||
AC_INIT([aria2],[1.14.2],[t-tujikawa@users.sourceforge.net],[aria2],[http://aria2.sourceforge.net/])
|
AC_INIT([aria2],[1.14.2],[t-tujikawa@users.sourceforge.net],[aria2],[http://aria2.sourceforge.net/])
|
||||||
|
LT_INIT()
|
||||||
AC_CANONICAL_HOST
|
AC_CANONICAL_HOST
|
||||||
AC_CANONICAL_TARGET
|
AC_CANONICAL_TARGET
|
||||||
AM_INIT_AUTOMAKE()
|
AM_INIT_AUTOMAKE()
|
||||||
|
@ -44,7 +46,6 @@ AC_PROG_CXX
|
||||||
AC_PROG_CC
|
AC_PROG_CC
|
||||||
AC_PROG_INSTALL
|
AC_PROG_INSTALL
|
||||||
AC_PROG_MKDIR_P
|
AC_PROG_MKDIR_P
|
||||||
AC_PROG_RANLIB
|
|
||||||
AC_PROG_YACC
|
AC_PROG_YACC
|
||||||
|
|
||||||
AC_CHECK_TOOL([AR], [ar], [:])
|
AC_CHECK_TOOL([AR], [ar], [:])
|
||||||
|
@ -526,6 +527,10 @@ if test "x$have_option_const_name" = "xyes"; then
|
||||||
AC_DEFINE([HAVE_OPTION_CONST_NAME], [1], [Define 1 if struct option.name is const char*])
|
AC_DEFINE([HAVE_OPTION_CONST_NAME], [1], [Define 1 if struct option.name is const char*])
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
AC_CONFIG_SUBDIRS([deps/wslay])
|
||||||
|
LIBS="\$(top_builddir)/deps/wslay/lib/libwslay.la $LIBS"
|
||||||
|
CPPFLAGS="-I\$(top_builddir)/deps/wslay/lib/includes $CPPFLAGS"
|
||||||
|
|
||||||
AC_CONFIG_FILES([Makefile
|
AC_CONFIG_FILES([Makefile
|
||||||
src/Makefile
|
src/Makefile
|
||||||
test/Makefile
|
test/Makefile
|
||||||
|
@ -533,7 +538,8 @@ AC_CONFIG_FILES([Makefile
|
||||||
intl/Makefile
|
intl/Makefile
|
||||||
lib/Makefile
|
lib/Makefile
|
||||||
doc/Makefile
|
doc/Makefile
|
||||||
doc/ru/Makefile])
|
doc/ru/Makefile
|
||||||
|
deps/Makefile])
|
||||||
AC_OUTPUT
|
AC_OUTPUT
|
||||||
|
|
||||||
echo " "
|
echo " "
|
||||||
|
|
1
deps/Makefile.am
vendored
Normal file
1
deps/Makefile.am
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
SUBDIRS = wslay
|
101
src/AbstractHttpServerResponseCommand.cc
Normal file
101
src/AbstractHttpServerResponseCommand.cc
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#include "AbstractHttpServerResponseCommand.h"
|
||||||
|
#include "SocketCore.h"
|
||||||
|
#include "DownloadEngine.h"
|
||||||
|
#include "HttpServer.h"
|
||||||
|
#include "Logger.h"
|
||||||
|
#include "LogFactory.h"
|
||||||
|
#include "HttpServerCommand.h"
|
||||||
|
#include "RequestGroupMan.h"
|
||||||
|
#include "RecoverableException.h"
|
||||||
|
#include "wallclock.h"
|
||||||
|
#include "util.h"
|
||||||
|
#include "fmt.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
AbstractHttpServerResponseCommand::AbstractHttpServerResponseCommand
|
||||||
|
(cuid_t cuid,
|
||||||
|
const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e,
|
||||||
|
const SharedHandle<SocketCore>& socket)
|
||||||
|
: Command(cuid),
|
||||||
|
e_(e),
|
||||||
|
socket_(socket),
|
||||||
|
httpServer_(httpServer)
|
||||||
|
{
|
||||||
|
setStatus(Command::STATUS_ONESHOT_REALTIME);
|
||||||
|
e_->addSocketForWriteCheck(socket_, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
AbstractHttpServerResponseCommand::~AbstractHttpServerResponseCommand()
|
||||||
|
{
|
||||||
|
e_->deleteSocketForWriteCheck(socket_, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AbstractHttpServerResponseCommand::execute()
|
||||||
|
{
|
||||||
|
if(e_->getRequestGroupMan()->downloadFinished() || e_->isHaltRequested()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
httpServer_->sendResponse();
|
||||||
|
} catch(RecoverableException& e) {
|
||||||
|
A2_LOG_INFO_EX
|
||||||
|
(fmt("CUID#%lld - Error occurred while transmitting response body.",
|
||||||
|
getCuid()),
|
||||||
|
e);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if(httpServer_->sendBufferIsEmpty()) {
|
||||||
|
A2_LOG_INFO(fmt("CUID#%lld - HttpServer: all response transmitted.",
|
||||||
|
getCuid()));
|
||||||
|
afterSend(httpServer_, e_);
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
if(timeoutTimer_.difference(global::wallclock()) >= 10) {
|
||||||
|
A2_LOG_INFO(fmt("CUID#%lld - HttpServer: Timeout while trasmitting"
|
||||||
|
" response.",
|
||||||
|
getCuid()));
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
e_->addCommand(this);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace aria2
|
75
src/AbstractHttpServerResponseCommand.h
Normal file
75
src/AbstractHttpServerResponseCommand.h
Normal file
|
@ -0,0 +1,75 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#ifndef D_ABSTRACT_HTTP_SERVER_RESPONSE_COMMAND_H
|
||||||
|
#define D_ABSTRACT_HTTP_SERVER_RESPONSE_COMMAND_H
|
||||||
|
|
||||||
|
#include "Command.h"
|
||||||
|
#include "SharedHandle.h"
|
||||||
|
#include "TimerA2.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
class DownloadEngine;
|
||||||
|
class SocketCore;
|
||||||
|
class HttpServer;
|
||||||
|
|
||||||
|
class AbstractHttpServerResponseCommand : public Command {
|
||||||
|
private:
|
||||||
|
DownloadEngine* e_;
|
||||||
|
SharedHandle<SocketCore> socket_;
|
||||||
|
SharedHandle<HttpServer> httpServer_;
|
||||||
|
Timer timeoutTimer_;
|
||||||
|
protected:
|
||||||
|
DownloadEngine* getDownloadEngine()
|
||||||
|
{
|
||||||
|
return e_;
|
||||||
|
}
|
||||||
|
// Called after content body is completely sent.
|
||||||
|
virtual void afterSend(const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e) = 0;
|
||||||
|
public:
|
||||||
|
AbstractHttpServerResponseCommand(cuid_t cuid,
|
||||||
|
const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e,
|
||||||
|
const SharedHandle<SocketCore>& socket);
|
||||||
|
|
||||||
|
virtual ~AbstractHttpServerResponseCommand();
|
||||||
|
|
||||||
|
virtual bool execute();
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace aria2
|
||||||
|
|
||||||
|
#endif // D_ABSTRACT_HTTP_SERVER_RESPONSE_COMMAND_H
|
|
@ -63,6 +63,8 @@
|
||||||
#include "PieceStatMan.h"
|
#include "PieceStatMan.h"
|
||||||
#include "wallclock.h"
|
#include "wallclock.h"
|
||||||
#include "bitfield.h"
|
#include "bitfield.h"
|
||||||
|
#include "SingletonHolder.h"
|
||||||
|
#include "Notifier.h"
|
||||||
#ifdef ENABLE_BITTORRENT
|
#ifdef ENABLE_BITTORRENT
|
||||||
# include "bittorrent_helper.h"
|
# include "bittorrent_helper.h"
|
||||||
#endif // ENABLE_BITTORRENT
|
#endif // ENABLE_BITTORRENT
|
||||||
|
@ -479,6 +481,9 @@ void DefaultPieceStorage::completePiece(const SharedHandle<Piece>& piece)
|
||||||
if(!torrentAttrs->metadata.empty()) {
|
if(!torrentAttrs->metadata.empty()) {
|
||||||
util::executeHookByOptName(downloadContext_->getOwnerRequestGroup(),
|
util::executeHookByOptName(downloadContext_->getOwnerRequestGroup(),
|
||||||
option_, PREF_ON_BT_DOWNLOAD_COMPLETE);
|
option_, PREF_ON_BT_DOWNLOAD_COMPLETE);
|
||||||
|
SingletonHolder<Notifier>::instance()->
|
||||||
|
notifyDownloadEvent(Notifier::ON_BT_DOWNLOAD_COMPLETE,
|
||||||
|
downloadContext_->getOwnerRequestGroup());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif // ENABLE_BITTORRENT
|
#endif // ENABLE_BITTORRENT
|
||||||
|
|
|
@ -212,6 +212,20 @@ void HttpServer::feedResponse(const std::string& status,
|
||||||
socketBuffer_.pushStr(text);
|
socketBuffer_.pushStr(text);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HttpServer::feedUpgradeResponse(const std::string& protocol,
|
||||||
|
const std::string& headers)
|
||||||
|
{
|
||||||
|
std::string header= fmt("HTTP/1.1 101 Switching Protocols\r\n"
|
||||||
|
"Upgrade: %s\r\n"
|
||||||
|
"Connection: Upgrade\r\n"
|
||||||
|
"%s"
|
||||||
|
"\r\n",
|
||||||
|
protocol.c_str(),
|
||||||
|
headers.c_str());
|
||||||
|
A2_LOG_DEBUG(fmt("HTTP Server sends upgrade response:\n%s", header.c_str()));
|
||||||
|
socketBuffer_.pushStr(header);
|
||||||
|
}
|
||||||
|
|
||||||
ssize_t HttpServer::sendResponse()
|
ssize_t HttpServer::sendResponse()
|
||||||
{
|
{
|
||||||
return socketBuffer_.send();
|
return socketBuffer_.send();
|
||||||
|
|
|
@ -91,6 +91,12 @@ public:
|
||||||
std::string& text,
|
std::string& text,
|
||||||
const std::string& contentType);
|
const std::string& contentType);
|
||||||
|
|
||||||
|
// Feeds "101 Switching Protocols" response. The |protocol| will
|
||||||
|
// appear in Upgrade header field. The |headers| is zero or more
|
||||||
|
// lines of HTTP header field and each line must end with "\r\n".
|
||||||
|
void feedUpgradeResponse(const std::string& protocol,
|
||||||
|
const std::string& headers);
|
||||||
|
|
||||||
bool authenticate();
|
bool authenticate();
|
||||||
|
|
||||||
void setUsernamePassword
|
void setUsernamePassword
|
||||||
|
@ -129,6 +135,11 @@ public:
|
||||||
{
|
{
|
||||||
allowOrigin_ = allowOrigin;
|
allowOrigin_ = allowOrigin;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const SharedHandle<SocketCore>& getSocket() const
|
||||||
|
{
|
||||||
|
return socket_;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
|
@ -86,21 +86,6 @@ HttpServerBodyCommand::~HttpServerBodyCommand()
|
||||||
e_->deleteSocketForReadCheck(socket_, this);
|
e_->deleteSocketForReadCheck(socket_, this);
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace {
|
|
||||||
rpc::RpcResponse
|
|
||||||
createJsonRpcErrorResponse
|
|
||||||
(int code,
|
|
||||||
const std::string& msg,
|
|
||||||
const SharedHandle<ValueBase>& id)
|
|
||||||
{
|
|
||||||
SharedHandle<Dict> params = Dict::g();
|
|
||||||
params->put("code", Integer::g(code));
|
|
||||||
params->put("message", msg);
|
|
||||||
rpc::RpcResponse res(code, params, id);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::string getJsonRpcContentType(bool script)
|
std::string getJsonRpcContentType(bool script)
|
||||||
{
|
{
|
||||||
|
@ -156,42 +141,6 @@ void HttpServerBodyCommand::addHttpServerResponseCommand()
|
||||||
e_->setNoWait(true);
|
e_->setNoWait(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
rpc::RpcResponse
|
|
||||||
HttpServerBodyCommand::processJsonRpcRequest(const Dict* jsondict)
|
|
||||||
{
|
|
||||||
|
|
||||||
SharedHandle<ValueBase> id = jsondict->get("id");
|
|
||||||
if(!id) {
|
|
||||||
return createJsonRpcErrorResponse(-32600, "Invalid Request.", Null::g());
|
|
||||||
}
|
|
||||||
const String* methodName = downcast<String>(jsondict->get("method"));
|
|
||||||
if(!methodName) {
|
|
||||||
return createJsonRpcErrorResponse(-32600, "Invalid Request.", id);
|
|
||||||
}
|
|
||||||
SharedHandle<List> params;
|
|
||||||
const SharedHandle<ValueBase>& tempParams = jsondict->get("params");
|
|
||||||
if(downcast<List>(tempParams)) {
|
|
||||||
params = static_pointer_cast<List>(tempParams);
|
|
||||||
} else if(!tempParams) {
|
|
||||||
params = List::g();
|
|
||||||
} else {
|
|
||||||
// TODO No support for Named params
|
|
||||||
return createJsonRpcErrorResponse(-32602, "Invalid params.", id);
|
|
||||||
}
|
|
||||||
rpc::RpcRequest req(methodName->s(), params, id);
|
|
||||||
req.jsonRpc = true;
|
|
||||||
SharedHandle<rpc::RpcMethod> method;
|
|
||||||
try {
|
|
||||||
method = rpc::RpcMethodFactory::create(req.methodName);
|
|
||||||
} catch(RecoverableException& e) {
|
|
||||||
A2_LOG_INFO_EX(EX_EXCEPTION_CAUGHT, e);
|
|
||||||
return createJsonRpcErrorResponse(-32601, "Method not found.", id);
|
|
||||||
}
|
|
||||||
A2_LOG_INFO(fmt("Executing RPC method %s", req.methodName.c_str()));
|
|
||||||
rpc::RpcResponse res = method->execute(req, e_);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool HttpServerBodyCommand::execute()
|
bool HttpServerBodyCommand::execute()
|
||||||
{
|
{
|
||||||
if(e_->getRequestGroupMan()->downloadFinished() || e_->isHaltRequested()) {
|
if(e_->getRequestGroupMan()->downloadFinished() || e_->isHaltRequested()) {
|
||||||
|
@ -242,13 +191,14 @@ bool HttpServerBodyCommand::execute()
|
||||||
getCuid()),
|
getCuid()),
|
||||||
e);
|
e);
|
||||||
rpc::RpcResponse res
|
rpc::RpcResponse res
|
||||||
(createJsonRpcErrorResponse(-32700, "Parse error.", Null::g()));
|
(rpc::createJsonRpcErrorResponse(-32700, "Parse error.",
|
||||||
|
Null::g()));
|
||||||
sendJsonRpcResponse(res, callback);
|
sendJsonRpcResponse(res, callback);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
const Dict* jsondict = downcast<Dict>(json);
|
const Dict* jsondict = downcast<Dict>(json);
|
||||||
if(jsondict) {
|
if(jsondict) {
|
||||||
rpc::RpcResponse res = processJsonRpcRequest(jsondict);
|
rpc::RpcResponse res = rpc::processJsonRpcRequest(jsondict, e_);
|
||||||
sendJsonRpcResponse(res, callback);
|
sendJsonRpcResponse(res, callback);
|
||||||
} else {
|
} else {
|
||||||
const List* jsonlist = downcast<List>(json);
|
const List* jsonlist = downcast<List>(json);
|
||||||
|
@ -259,14 +209,14 @@ bool HttpServerBodyCommand::execute()
|
||||||
eoi = jsonlist->end(); i != eoi; ++i) {
|
eoi = jsonlist->end(); i != eoi; ++i) {
|
||||||
const Dict* jsondict = downcast<Dict>(*i);
|
const Dict* jsondict = downcast<Dict>(*i);
|
||||||
if(jsondict) {
|
if(jsondict) {
|
||||||
rpc::RpcResponse r = processJsonRpcRequest(jsondict);
|
rpc::RpcResponse r = rpc::processJsonRpcRequest(jsondict, e_);
|
||||||
results.push_back(r);
|
results.push_back(r);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
sendJsonRpcBatchResponse(results, callback);
|
sendJsonRpcBatchResponse(results, callback);
|
||||||
} else {
|
} else {
|
||||||
rpc::RpcResponse res
|
rpc::RpcResponse res
|
||||||
(createJsonRpcErrorResponse
|
(rpc::createJsonRpcErrorResponse
|
||||||
(-32600, "Invalid Request.", Null::g()));
|
(-32600, "Invalid Request.", Null::g()));
|
||||||
sendJsonRpcResponse(res, callback);
|
sendJsonRpcResponse(res, callback);
|
||||||
}
|
}
|
||||||
|
|
|
@ -65,7 +65,6 @@ private:
|
||||||
void sendJsonRpcBatchResponse
|
void sendJsonRpcBatchResponse
|
||||||
(const std::vector<rpc::RpcResponse>& results,
|
(const std::vector<rpc::RpcResponse>& results,
|
||||||
const std::string& callback);
|
const std::string& callback);
|
||||||
rpc::RpcResponse processJsonRpcRequest(const Dict* jsondict);
|
|
||||||
void addHttpServerResponseCommand();
|
void addHttpServerResponseCommand();
|
||||||
public:
|
public:
|
||||||
HttpServerBodyCommand(cuid_t cuid,
|
HttpServerBodyCommand(cuid_t cuid,
|
||||||
|
|
|
@ -43,6 +43,7 @@
|
||||||
#include "RequestGroupMan.h"
|
#include "RequestGroupMan.h"
|
||||||
#include "HttpServerBodyCommand.h"
|
#include "HttpServerBodyCommand.h"
|
||||||
#include "HttpServerResponseCommand.h"
|
#include "HttpServerResponseCommand.h"
|
||||||
|
#include "WebSocketResponseCommand.h"
|
||||||
#include "RecoverableException.h"
|
#include "RecoverableException.h"
|
||||||
#include "prefs.h"
|
#include "prefs.h"
|
||||||
#include "Option.h"
|
#include "Option.h"
|
||||||
|
@ -50,6 +51,9 @@
|
||||||
#include "wallclock.h"
|
#include "wallclock.h"
|
||||||
#include "fmt.h"
|
#include "fmt.h"
|
||||||
#include "SocketRecvBuffer.h"
|
#include "SocketRecvBuffer.h"
|
||||||
|
#include "MessageDigest.h"
|
||||||
|
#include "message_digest_helper.h"
|
||||||
|
#include "base64.h"
|
||||||
|
|
||||||
namespace aria2 {
|
namespace aria2 {
|
||||||
|
|
||||||
|
@ -104,6 +108,19 @@ void HttpServerCommand::checkSocketRecvBuffer()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Creates server's WebSocket accept key which will be sent in
|
||||||
|
// Sec-WebSocket-Accept header field. The |clientKey| is the value
|
||||||
|
// found in Sec-WebSocket-Key header field in the request.
|
||||||
|
std::string createWebSocketServerKey(const std::string& clientKey)
|
||||||
|
{
|
||||||
|
std::string src = clientKey;
|
||||||
|
src += "258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
|
||||||
|
unsigned char digest[20];
|
||||||
|
message_digest::digest(digest, sizeof(digest), MessageDigest::sha1(),
|
||||||
|
src.c_str(), src.size());
|
||||||
|
return base64::encode(&digest[0], &digest[sizeof(digest)]);
|
||||||
|
}
|
||||||
|
|
||||||
bool HttpServerCommand::execute()
|
bool HttpServerCommand::execute()
|
||||||
{
|
{
|
||||||
if(e_->getRequestGroupMan()->downloadFinished() || e_->isHaltRequested()) {
|
if(e_->getRequestGroupMan()->downloadFinished() || e_->isHaltRequested()) {
|
||||||
|
@ -133,20 +150,41 @@ bool HttpServerCommand::execute()
|
||||||
e_->setNoWait(true);
|
e_->setNoWait(true);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if(e_->getOption()->getAsInt(PREF_RPC_MAX_REQUEST_SIZE) <
|
const std::string& upgradeHd = header->find("upgrade");
|
||||||
httpServer_->getContentLength()) {
|
const std::string& connectionHd = header->find("connection");
|
||||||
A2_LOG_INFO
|
if(httpServer_->getRequestPath() == "/jsonrpc" &&
|
||||||
(fmt("Request too long. ContentLength=%lld."
|
httpServer_->getMethod() == "GET" &&
|
||||||
" See --rpc-max-request-size option to loose"
|
util::strieq(upgradeHd.begin(), upgradeHd.end(), "websocket") &&
|
||||||
" this limitation.",
|
util::strieq(connectionHd.begin(), connectionHd.end(), "upgrade") &&
|
||||||
static_cast<long long int>(httpServer_->getContentLength())));
|
header->find("sec-websocket-version") == "13" &&
|
||||||
|
header->defined("sec-websocket-key")) {
|
||||||
|
std::string serverKey =
|
||||||
|
createWebSocketServerKey(header->find("sec-websocket-key"));
|
||||||
|
httpServer_->feedUpgradeResponse("websocket",
|
||||||
|
fmt("Sec-WebSocket-Accept: %s\r\n",
|
||||||
|
serverKey.c_str()));
|
||||||
|
Command* command =
|
||||||
|
new rpc::WebSocketResponseCommand(getCuid(), httpServer_, e_,
|
||||||
|
socket_);
|
||||||
|
e_->addCommand(command);
|
||||||
|
e_->setNoWait(true);
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
if(e_->getOption()->getAsInt(PREF_RPC_MAX_REQUEST_SIZE) <
|
||||||
|
httpServer_->getContentLength()) {
|
||||||
|
A2_LOG_INFO
|
||||||
|
(fmt("Request too long. ContentLength=%lld."
|
||||||
|
" See --rpc-max-request-size option to loose"
|
||||||
|
" this limitation.",
|
||||||
|
static_cast<long long int>(httpServer_->getContentLength())));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
Command* command = new HttpServerBodyCommand(getCuid(), httpServer_, e_,
|
||||||
|
socket_);
|
||||||
|
e_->addCommand(command);
|
||||||
|
e_->setNoWait(true);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
Command* command = new HttpServerBodyCommand(getCuid(), httpServer_, e_,
|
|
||||||
socket_);
|
|
||||||
e_->addCommand(command);
|
|
||||||
e_->setNoWait(true);
|
|
||||||
return true;
|
|
||||||
} else {
|
} else {
|
||||||
if(timeoutTimer_.difference(global::wallclock()) >= 30) {
|
if(timeoutTimer_.difference(global::wallclock()) >= 30) {
|
||||||
A2_LOG_INFO("HTTP request timeout.");
|
A2_LOG_INFO("HTTP request timeout.");
|
||||||
|
|
|
@ -40,9 +40,6 @@
|
||||||
#include "LogFactory.h"
|
#include "LogFactory.h"
|
||||||
#include "HttpServerCommand.h"
|
#include "HttpServerCommand.h"
|
||||||
#include "RequestGroupMan.h"
|
#include "RequestGroupMan.h"
|
||||||
#include "RecoverableException.h"
|
|
||||||
#include "wallclock.h"
|
|
||||||
#include "util.h"
|
|
||||||
#include "fmt.h"
|
#include "fmt.h"
|
||||||
|
|
||||||
namespace aria2 {
|
namespace aria2 {
|
||||||
|
@ -52,54 +49,22 @@ HttpServerResponseCommand::HttpServerResponseCommand
|
||||||
const SharedHandle<HttpServer>& httpServer,
|
const SharedHandle<HttpServer>& httpServer,
|
||||||
DownloadEngine* e,
|
DownloadEngine* e,
|
||||||
const SharedHandle<SocketCore>& socket)
|
const SharedHandle<SocketCore>& socket)
|
||||||
: Command(cuid),
|
: AbstractHttpServerResponseCommand(cuid, httpServer, e, socket)
|
||||||
e_(e),
|
{}
|
||||||
socket_(socket),
|
|
||||||
httpServer_(httpServer)
|
|
||||||
{
|
|
||||||
setStatus(Command::STATUS_ONESHOT_REALTIME);
|
|
||||||
e_->addSocketForWriteCheck(socket_, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpServerResponseCommand::~HttpServerResponseCommand()
|
HttpServerResponseCommand::~HttpServerResponseCommand()
|
||||||
{
|
{}
|
||||||
e_->deleteSocketForWriteCheck(socket_, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool HttpServerResponseCommand::execute()
|
void HttpServerResponseCommand::afterSend
|
||||||
|
(const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e)
|
||||||
{
|
{
|
||||||
if(e_->getRequestGroupMan()->downloadFinished() || e_->isHaltRequested()) {
|
if(httpServer->supportsPersistentConnection()) {
|
||||||
return true;
|
A2_LOG_INFO(fmt("CUID#%lld - Persist connection.",
|
||||||
}
|
|
||||||
try {
|
|
||||||
httpServer_->sendResponse();
|
|
||||||
} catch(RecoverableException& e) {
|
|
||||||
A2_LOG_INFO_EX
|
|
||||||
(fmt("CUID#%lld - Error occurred while transmitting response body.",
|
|
||||||
getCuid()),
|
|
||||||
e);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if(httpServer_->sendBufferIsEmpty()) {
|
|
||||||
A2_LOG_INFO(fmt("CUID#%lld - HttpServer: all response transmitted.",
|
|
||||||
getCuid()));
|
getCuid()));
|
||||||
if(httpServer_->supportsPersistentConnection()) {
|
e->addCommand
|
||||||
A2_LOG_INFO(fmt("CUID#%lld - Persist connection.",
|
(new HttpServerCommand(getCuid(), httpServer, e,
|
||||||
getCuid()));
|
httpServer->getSocket()));
|
||||||
e_->addCommand
|
|
||||||
(new HttpServerCommand(getCuid(), httpServer_, e_, socket_));
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
if(timeoutTimer_.difference(global::wallclock()) >= 10) {
|
|
||||||
A2_LOG_INFO(fmt("CUID#%lld - HttpServer: Timeout while trasmitting"
|
|
||||||
" response.",
|
|
||||||
getCuid()));
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
e_->addCommand(this);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -35,22 +35,14 @@
|
||||||
#ifndef D_HTTP_SERVER_RESPONSE_COMMAND_H
|
#ifndef D_HTTP_SERVER_RESPONSE_COMMAND_H
|
||||||
#define D_HTTP_SERVER_RESPONSE_COMMAND_H
|
#define D_HTTP_SERVER_RESPONSE_COMMAND_H
|
||||||
|
|
||||||
#include "Command.h"
|
#include "AbstractHttpServerResponseCommand.h"
|
||||||
#include "SharedHandle.h"
|
|
||||||
#include "TimerA2.h"
|
|
||||||
|
|
||||||
namespace aria2 {
|
namespace aria2 {
|
||||||
|
|
||||||
class DownloadEngine;
|
class HttpServerResponseCommand : public AbstractHttpServerResponseCommand {
|
||||||
class SocketCore;
|
protected:
|
||||||
class HttpServer;
|
virtual void afterSend(const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e);
|
||||||
class HttpServerResponseCommand : public Command {
|
|
||||||
private:
|
|
||||||
DownloadEngine* e_;
|
|
||||||
SharedHandle<SocketCore> socket_;
|
|
||||||
SharedHandle<HttpServer> httpServer_;
|
|
||||||
Timer timeoutTimer_;
|
|
||||||
public:
|
public:
|
||||||
HttpServerResponseCommand(cuid_t cuid,
|
HttpServerResponseCommand(cuid_t cuid,
|
||||||
const SharedHandle<HttpServer>& httpServer,
|
const SharedHandle<HttpServer>& httpServer,
|
||||||
|
@ -58,8 +50,6 @@ public:
|
||||||
const SharedHandle<SocketCore>& socket);
|
const SharedHandle<SocketCore>& socket);
|
||||||
|
|
||||||
virtual ~HttpServerResponseCommand();
|
virtual ~HttpServerResponseCommand();
|
||||||
|
|
||||||
virtual bool execute();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
|
@ -208,6 +208,8 @@ SRCS = Socket.h\
|
||||||
HttpListenCommand.cc HttpListenCommand.h\
|
HttpListenCommand.cc HttpListenCommand.h\
|
||||||
HttpServerCommand.cc HttpServerCommand.h\
|
HttpServerCommand.cc HttpServerCommand.h\
|
||||||
HttpServerResponseCommand.cc HttpServerResponseCommand.h\
|
HttpServerResponseCommand.cc HttpServerResponseCommand.h\
|
||||||
|
AbstractHttpServerResponseCommand.cc \
|
||||||
|
AbstractHttpServerResponseCommand.h \
|
||||||
HttpServer.cc HttpServer.h\
|
HttpServer.cc HttpServer.h\
|
||||||
StreamPieceSelector.h\
|
StreamPieceSelector.h\
|
||||||
DefaultStreamPieceSelector.cc DefaultStreamPieceSelector.h\
|
DefaultStreamPieceSelector.cc DefaultStreamPieceSelector.h\
|
||||||
|
@ -224,7 +226,12 @@ SRCS = Socket.h\
|
||||||
paramed_string.cc paramed_string.h\
|
paramed_string.cc paramed_string.h\
|
||||||
rpc_helper.cc rpc_helper.h\
|
rpc_helper.cc rpc_helper.h\
|
||||||
WatchProcessCommand.cc WatchProcessCommand.h\
|
WatchProcessCommand.cc WatchProcessCommand.h\
|
||||||
UnknownOptionException.cc UnknownOptionException.h
|
UnknownOptionException.cc UnknownOptionException.h\
|
||||||
|
WebSocketSession.cc WebSocketSession.h\
|
||||||
|
WebSocketSessionMan.cc WebSocketSessionMan.h\
|
||||||
|
WebSocketResponseCommand.cc WebSocketResponseCommand.h\
|
||||||
|
WebSocketInteractionCommand.cc WebSocketInteractionCommand.h\
|
||||||
|
Notifier.cc Notifier.h
|
||||||
|
|
||||||
if MINGW_BUILD
|
if MINGW_BUILD
|
||||||
SRCS += WinConsoleFile.cc WinConsoleFile.h
|
SRCS += WinConsoleFile.cc WinConsoleFile.h
|
||||||
|
|
|
@ -61,6 +61,9 @@
|
||||||
#include "SocketCore.h"
|
#include "SocketCore.h"
|
||||||
#include "OutputFile.h"
|
#include "OutputFile.h"
|
||||||
#include "UriListParser.h"
|
#include "UriListParser.h"
|
||||||
|
#include "SingletonHolder.h"
|
||||||
|
#include "Notifier.h"
|
||||||
|
#include "WebSocketSessionMan.h"
|
||||||
#ifdef ENABLE_SSL
|
#ifdef ENABLE_SSL
|
||||||
# include "TLSContext.h"
|
# include "TLSContext.h"
|
||||||
#endif // ENABLE_SSL
|
#endif // ENABLE_SSL
|
||||||
|
@ -167,6 +170,13 @@ error_code::Value MultiUrlRequestInfo::execute()
|
||||||
{
|
{
|
||||||
error_code::Value returnValue = error_code::FINISHED;
|
error_code::Value returnValue = error_code::FINISHED;
|
||||||
try {
|
try {
|
||||||
|
SharedHandle<rpc::WebSocketSessionMan> wsSessionMan;
|
||||||
|
if(option_->getAsBool(PREF_ENABLE_RPC)) {
|
||||||
|
wsSessionMan.reset(new rpc::WebSocketSessionMan());
|
||||||
|
}
|
||||||
|
Notifier notifier(wsSessionMan);
|
||||||
|
SingletonHolder<Notifier>::instance(¬ifier);
|
||||||
|
|
||||||
DownloadEngineHandle e =
|
DownloadEngineHandle e =
|
||||||
DownloadEngineFactory().newDownloadEngine(option_.get(), requestGroups_);
|
DownloadEngineFactory().newDownloadEngine(option_.get(), requestGroups_);
|
||||||
|
|
||||||
|
@ -290,6 +300,7 @@ error_code::Value MultiUrlRequestInfo::execute()
|
||||||
}
|
}
|
||||||
A2_LOG_ERROR_EX(EX_EXCEPTION_CAUGHT, e);
|
A2_LOG_ERROR_EX(EX_EXCEPTION_CAUGHT, e);
|
||||||
}
|
}
|
||||||
|
SingletonHolder<Notifier>::instance(0);
|
||||||
#ifdef SIGHUP
|
#ifdef SIGHUP
|
||||||
util::setGlobalSignalHandler(SIGHUP, SIG_DFL, 0);
|
util::setGlobalSignalHandler(SIGHUP, SIG_DFL, 0);
|
||||||
#endif // SIGHUP
|
#endif // SIGHUP
|
||||||
|
|
77
src/Notifier.cc
Normal file
77
src/Notifier.cc
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#include "Notifier.h"
|
||||||
|
#include "RequestGroup.h"
|
||||||
|
#include "WebSocketSessionMan.h"
|
||||||
|
#include "LogFactory.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
Notifier::Notifier(const SharedHandle<rpc::WebSocketSessionMan>& wsSessionMan)
|
||||||
|
: wsSessionMan_(wsSessionMan)
|
||||||
|
{}
|
||||||
|
|
||||||
|
Notifier::~Notifier() {}
|
||||||
|
|
||||||
|
void Notifier::addWebSocketSession
|
||||||
|
(const SharedHandle<rpc::WebSocketSession>& wsSession)
|
||||||
|
{
|
||||||
|
A2_LOG_DEBUG("WebSocket session added.");
|
||||||
|
wsSessionMan_->addSession(wsSession);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Notifier::removeWebSocketSession
|
||||||
|
(const SharedHandle<rpc::WebSocketSession>& wsSession)
|
||||||
|
{
|
||||||
|
A2_LOG_DEBUG("WebSocket session removed.");
|
||||||
|
wsSessionMan_->removeSession(wsSession);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string Notifier::ON_DOWNLOAD_START = "onDownloadStart";
|
||||||
|
const std::string Notifier::ON_DOWNLOAD_PAUSE = "onDownloadPause";
|
||||||
|
const std::string Notifier::ON_DOWNLOAD_STOP = "onDownloadStop";
|
||||||
|
const std::string Notifier::ON_DOWNLOAD_COMPLETE = "onDownloadComplete";
|
||||||
|
const std::string Notifier::ON_DOWNLOAD_ERROR = "onDownloadError";
|
||||||
|
const std::string Notifier::ON_BT_DOWNLOAD_COMPLETE = "onBtDownloadComplete";
|
||||||
|
|
||||||
|
void Notifier::notifyDownloadEvent
|
||||||
|
(const std::string& event, const RequestGroup* group)
|
||||||
|
{
|
||||||
|
if(wsSessionMan_) {
|
||||||
|
wsSessionMan_->addNotification(event, group);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace aria2
|
82
src/Notifier.h
Normal file
82
src/Notifier.h
Normal file
|
@ -0,0 +1,82 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#ifndef D_NOTIFIER_H
|
||||||
|
#define D_NOTIFIER_H
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
|
#include "SharedHandle.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
class RequestGroup;
|
||||||
|
class Option;
|
||||||
|
class Pref;
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
class WebSocketSessionMan;
|
||||||
|
class WebSocketSession;
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
class Notifier {
|
||||||
|
public:
|
||||||
|
// The string constants for download events.
|
||||||
|
static const std::string ON_DOWNLOAD_START;
|
||||||
|
static const std::string ON_DOWNLOAD_PAUSE;
|
||||||
|
static const std::string ON_DOWNLOAD_STOP;
|
||||||
|
static const std::string ON_DOWNLOAD_COMPLETE;
|
||||||
|
static const std::string ON_DOWNLOAD_ERROR;
|
||||||
|
static const std::string ON_BT_DOWNLOAD_COMPLETE;
|
||||||
|
|
||||||
|
Notifier(const SharedHandle<rpc::WebSocketSessionMan>& wsSessionMan);
|
||||||
|
~Notifier();
|
||||||
|
void addWebSocketSession(const SharedHandle<rpc::WebSocketSession>& wsSes);
|
||||||
|
void removeWebSocketSession(const SharedHandle<rpc::WebSocketSession>& wsSes);
|
||||||
|
// Notifies the download event to all listeners.
|
||||||
|
void notifyDownloadEvent(const std::string& event, const RequestGroup* group);
|
||||||
|
|
||||||
|
void notifyDownloadEvent(const std::string& event,
|
||||||
|
const SharedHandle<RequestGroup>& group)
|
||||||
|
{
|
||||||
|
notifyDownloadEvent(event, group.get());
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
SharedHandle<rpc::WebSocketSessionMan> wsSessionMan_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace aria2
|
||||||
|
|
||||||
|
#endif // D_NOTIFIER_H
|
|
@ -76,6 +76,8 @@
|
||||||
#include "OutputFile.h"
|
#include "OutputFile.h"
|
||||||
#include "download_helper.h"
|
#include "download_helper.h"
|
||||||
#include "UriListParser.h"
|
#include "UriListParser.h"
|
||||||
|
#include "SingletonHolder.h"
|
||||||
|
#include "Notifier.h"
|
||||||
|
|
||||||
namespace aria2 {
|
namespace aria2 {
|
||||||
|
|
||||||
|
@ -252,6 +254,16 @@ bool RequestGroupMan::removeReservedGroup(a2_gid_t gid)
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
|
void notifyDownloadEvent
|
||||||
|
(const std::string& event, const SharedHandle<RequestGroup>& group)
|
||||||
|
{
|
||||||
|
SingletonHolder<Notifier>::instance()->notifyDownloadEvent(event, group);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
void executeStopHook
|
void executeStopHook
|
||||||
(const SharedHandle<RequestGroup>& group,
|
(const SharedHandle<RequestGroup>& group,
|
||||||
const Option* option,
|
const Option* option,
|
||||||
|
@ -267,6 +279,14 @@ void executeStopHook
|
||||||
} else if(!option->blank(PREF_ON_DOWNLOAD_STOP)) {
|
} else if(!option->blank(PREF_ON_DOWNLOAD_STOP)) {
|
||||||
util::executeHookByOptName(group, option, PREF_ON_DOWNLOAD_STOP);
|
util::executeHookByOptName(group, option, PREF_ON_DOWNLOAD_STOP);
|
||||||
}
|
}
|
||||||
|
if(result == error_code::FINISHED) {
|
||||||
|
notifyDownloadEvent(Notifier::ON_DOWNLOAD_COMPLETE, group);
|
||||||
|
} else if(result != error_code::IN_PROGRESS &&
|
||||||
|
result != error_code::REMOVED) {
|
||||||
|
notifyDownloadEvent(Notifier::ON_DOWNLOAD_ERROR, group);
|
||||||
|
} else {
|
||||||
|
notifyDownloadEvent(Notifier::ON_DOWNLOAD_STOP, group);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
@ -353,8 +373,9 @@ public:
|
||||||
reservedGroups_.push_front(group);
|
reservedGroups_.push_front(group);
|
||||||
group->releaseRuntimeResource(e_);
|
group->releaseRuntimeResource(e_);
|
||||||
group->setForceHaltRequested(false);
|
group->setForceHaltRequested(false);
|
||||||
util::executeHookByOptName
|
util::executeHookByOptName(group, e_->getOption(),
|
||||||
(group, e_->getOption(), PREF_ON_DOWNLOAD_PAUSE);
|
PREF_ON_DOWNLOAD_PAUSE);
|
||||||
|
notifyDownloadEvent(Notifier::ON_DOWNLOAD_PAUSE, group);
|
||||||
// TODO Should we have to prepend spend uris to remaining uris
|
// TODO Should we have to prepend spend uris to remaining uris
|
||||||
// in case PREF_REUSE_URI is disabed?
|
// in case PREF_REUSE_URI is disabed?
|
||||||
} else {
|
} else {
|
||||||
|
@ -536,8 +557,9 @@ void RequestGroupMan::fillRequestGroupFromReserver(DownloadEngine* e)
|
||||||
requestGroups_.push_back(groupToAdd);
|
requestGroups_.push_back(groupToAdd);
|
||||||
requestQueueCheck();
|
requestQueueCheck();
|
||||||
}
|
}
|
||||||
util::executeHookByOptName
|
util::executeHookByOptName(groupToAdd, e->getOption(),
|
||||||
(groupToAdd, e->getOption(), PREF_ON_DOWNLOAD_START);
|
PREF_ON_DOWNLOAD_START);
|
||||||
|
notifyDownloadEvent(Notifier::ON_DOWNLOAD_START, groupToAdd);
|
||||||
}
|
}
|
||||||
if(!temp.empty()) {
|
if(!temp.empty()) {
|
||||||
reservedGroups_.insert(reservedGroups_.begin(), temp.begin(), temp.end());
|
reservedGroups_.insert(reservedGroups_.begin(), temp.begin(), temp.end());
|
||||||
|
|
|
@ -40,25 +40,25 @@ namespace aria2 {
|
||||||
template<typename T>
|
template<typename T>
|
||||||
class SingletonHolder {
|
class SingletonHolder {
|
||||||
private:
|
private:
|
||||||
static T instance_;
|
static T* instance_;
|
||||||
|
|
||||||
SingletonHolder() {}
|
SingletonHolder() {}
|
||||||
public:
|
public:
|
||||||
~SingletonHolder() {}
|
~SingletonHolder() {}
|
||||||
|
|
||||||
static T& instance()
|
static T* instance()
|
||||||
{
|
{
|
||||||
return instance_;
|
return instance_;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void instance(T& instance)
|
static void instance(T* instance)
|
||||||
{
|
{
|
||||||
instance_ = instance;
|
instance_ = instance;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
T SingletonHolder<T>::instance_;
|
T* SingletonHolder<T>::instance_ = 0;
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
||||||
|
|
111
src/WebSocketInteractionCommand.cc
Normal file
111
src/WebSocketInteractionCommand.cc
Normal file
|
@ -0,0 +1,111 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#include "WebSocketInteractionCommand.h"
|
||||||
|
#include "SocketCore.h"
|
||||||
|
#include "DownloadEngine.h"
|
||||||
|
#include "RequestGroupMan.h"
|
||||||
|
#include "WebSocketSession.h"
|
||||||
|
#include "Logger.h"
|
||||||
|
#include "LogFactory.h"
|
||||||
|
#include "fmt.h"
|
||||||
|
#include "SingletonHolder.h"
|
||||||
|
#include "Notifier.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
WebSocketInteractionCommand::WebSocketInteractionCommand
|
||||||
|
(cuid_t cuid,
|
||||||
|
const SharedHandle<WebSocketSession>& wsSession,
|
||||||
|
DownloadEngine* e,
|
||||||
|
const SharedHandle<SocketCore>& socket)
|
||||||
|
: Command(cuid),
|
||||||
|
e_(e),
|
||||||
|
socket_(socket),
|
||||||
|
writeCheck_(false),
|
||||||
|
wsSession_(wsSession)
|
||||||
|
{
|
||||||
|
SingletonHolder<Notifier>::instance()->addWebSocketSession(wsSession_);
|
||||||
|
e_->addSocketForReadCheck(socket_, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
WebSocketInteractionCommand::~WebSocketInteractionCommand()
|
||||||
|
{
|
||||||
|
e_->deleteSocketForReadCheck(socket_, this);
|
||||||
|
if(writeCheck_) {
|
||||||
|
e_->deleteSocketForWriteCheck(socket_, this);
|
||||||
|
}
|
||||||
|
SingletonHolder<Notifier>::instance()->removeWebSocketSession(wsSession_);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebSocketInteractionCommand::updateWriteCheck()
|
||||||
|
{
|
||||||
|
if(wsSession_->wantWrite()) {
|
||||||
|
if(!writeCheck_) {
|
||||||
|
writeCheck_ = true;
|
||||||
|
e_->addSocketForWriteCheck(socket_, this);
|
||||||
|
}
|
||||||
|
} else if(writeCheck_) {
|
||||||
|
writeCheck_ = false;
|
||||||
|
e_->deleteSocketForWriteCheck(socket_, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WebSocketInteractionCommand::execute()
|
||||||
|
{
|
||||||
|
if(e_->isHaltRequested()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if(wsSession_->onReadEvent() == -1 || wsSession_->onWriteEvent() == -1) {
|
||||||
|
if(wsSession_->closeSent() || wsSession_->closeReceived()) {
|
||||||
|
A2_LOG_INFO(fmt("CUID#%lld - WebSocket session terminated.", getCuid()));
|
||||||
|
} else {
|
||||||
|
A2_LOG_INFO(fmt("CUID#%lld - WebSocket session terminated"
|
||||||
|
" (Possibly due to EOF).", getCuid()));
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if(wsSession_->finish()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
updateWriteCheck();
|
||||||
|
e_->addCommand(this);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
} // namespace aria2
|
73
src/WebSocketInteractionCommand.h
Normal file
73
src/WebSocketInteractionCommand.h
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#ifndef D_WEB_SOCKET_INTERACTION_COMMAND_H
|
||||||
|
#define D_WEB_SOCKET_INTERACTION_COMMAND_H
|
||||||
|
|
||||||
|
#include "Command.h"
|
||||||
|
#include "SharedHandle.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
class DownloadEngine;
|
||||||
|
class SocketCore;
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
class WebSocketSession;
|
||||||
|
|
||||||
|
class WebSocketInteractionCommand : public Command {
|
||||||
|
private:
|
||||||
|
DownloadEngine* e_;
|
||||||
|
SharedHandle<SocketCore> socket_;
|
||||||
|
bool writeCheck_;
|
||||||
|
SharedHandle<WebSocketSession> wsSession_;
|
||||||
|
public:
|
||||||
|
WebSocketInteractionCommand(cuid_t cuid,
|
||||||
|
const SharedHandle<WebSocketSession>& wsSession,
|
||||||
|
DownloadEngine* e,
|
||||||
|
const SharedHandle<SocketCore>& socket);
|
||||||
|
|
||||||
|
virtual ~WebSocketInteractionCommand();
|
||||||
|
|
||||||
|
virtual bool execute();
|
||||||
|
|
||||||
|
void updateWriteCheck();
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
} // namespace aria2
|
||||||
|
|
||||||
|
#endif // D_WEB_SOCKET_INTERACTION_COMMAND_H
|
72
src/WebSocketResponseCommand.cc
Normal file
72
src/WebSocketResponseCommand.cc
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#include "WebSocketResponseCommand.h"
|
||||||
|
#include "SocketCore.h"
|
||||||
|
#include "DownloadEngine.h"
|
||||||
|
#include "HttpServer.h"
|
||||||
|
#include "WebSocketSession.h"
|
||||||
|
#include "WebSocketInteractionCommand.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
WebSocketResponseCommand::WebSocketResponseCommand
|
||||||
|
(cuid_t cuid,
|
||||||
|
const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e,
|
||||||
|
const SharedHandle<SocketCore>& socket)
|
||||||
|
: AbstractHttpServerResponseCommand(cuid, httpServer, e, socket)
|
||||||
|
{}
|
||||||
|
|
||||||
|
WebSocketResponseCommand::~WebSocketResponseCommand()
|
||||||
|
{}
|
||||||
|
|
||||||
|
void WebSocketResponseCommand::afterSend
|
||||||
|
(const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e)
|
||||||
|
{
|
||||||
|
SharedHandle<WebSocketSession> wsSession
|
||||||
|
(new WebSocketSession(httpServer->getSocket(), getDownloadEngine()));
|
||||||
|
WebSocketInteractionCommand* command =
|
||||||
|
new WebSocketInteractionCommand(getCuid(), wsSession, e,
|
||||||
|
wsSession->getSocket());
|
||||||
|
wsSession->setCommand(command);
|
||||||
|
e->addCommand(command);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
} // namespace aria2
|
61
src/WebSocketResponseCommand.h
Normal file
61
src/WebSocketResponseCommand.h
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#ifndef D_WEB_SOCKET_RESPONSE_COMMAND_H
|
||||||
|
#define D_WEB_SOCKET_RESPONSE_COMMAND_H
|
||||||
|
|
||||||
|
#include "AbstractHttpServerResponseCommand.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
class WebSocketResponseCommand : public AbstractHttpServerResponseCommand {
|
||||||
|
protected:
|
||||||
|
virtual void afterSend(const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e);
|
||||||
|
public:
|
||||||
|
WebSocketResponseCommand(cuid_t cuid,
|
||||||
|
const SharedHandle<HttpServer>& httpServer,
|
||||||
|
DownloadEngine* e,
|
||||||
|
const SharedHandle<SocketCore>& socket);
|
||||||
|
|
||||||
|
virtual ~WebSocketResponseCommand();
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
} // namespace aria2
|
||||||
|
|
||||||
|
#endif // D_WEB_SOCKET_RESPONSE_COMMAND_H
|
251
src/WebSocketSession.cc
Normal file
251
src/WebSocketSession.cc
Normal file
|
@ -0,0 +1,251 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#include "WebSocketSession.h"
|
||||||
|
|
||||||
|
#include <cerrno>
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
#include "SocketCore.h"
|
||||||
|
#include "LogFactory.h"
|
||||||
|
#include "RecoverableException.h"
|
||||||
|
#include "message.h"
|
||||||
|
#include "DownloadEngine.h"
|
||||||
|
#include "rpc_helper.h"
|
||||||
|
#include "RpcResponse.h"
|
||||||
|
#include "json.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
ssize_t sendCallback(wslay_event_context_ptr wsctx,
|
||||||
|
const uint8_t* data, size_t len, int flags,
|
||||||
|
void* userData)
|
||||||
|
{
|
||||||
|
WebSocketSession* session = reinterpret_cast<WebSocketSession*>(userData);
|
||||||
|
const SharedHandle<SocketCore>& socket = session->getSocket();
|
||||||
|
try {
|
||||||
|
ssize_t r = socket->writeData(data, len);
|
||||||
|
if(r == 0) {
|
||||||
|
if(socket->wantRead() || socket->wantWrite()) {
|
||||||
|
wslay_event_set_error(wsctx, WSLAY_ERR_WOULDBLOCK);
|
||||||
|
} else {
|
||||||
|
wslay_event_set_error(wsctx, WSLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
}
|
||||||
|
r = -1;
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
} catch(RecoverableException& e) {
|
||||||
|
A2_LOG_DEBUG_EX(EX_EXCEPTION_CAUGHT, e);
|
||||||
|
wslay_event_set_error(wsctx, WSLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
ssize_t recvCallback(wslay_event_context_ptr wsctx,
|
||||||
|
uint8_t* buf, size_t len, int flags,
|
||||||
|
void* userData)
|
||||||
|
{
|
||||||
|
WebSocketSession* session = reinterpret_cast<WebSocketSession*>(userData);
|
||||||
|
const SharedHandle<SocketCore>& socket = session->getSocket();
|
||||||
|
try {
|
||||||
|
ssize_t r;
|
||||||
|
socket->readData(buf, len);
|
||||||
|
if(len == 0) {
|
||||||
|
if(socket->wantRead() || socket->wantWrite()) {
|
||||||
|
wslay_event_set_error(wsctx, WSLAY_ERR_WOULDBLOCK);
|
||||||
|
} else {
|
||||||
|
wslay_event_set_error(wsctx, WSLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
}
|
||||||
|
r = -1;
|
||||||
|
} else {
|
||||||
|
r = len;
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
} catch(RecoverableException& e) {
|
||||||
|
A2_LOG_DEBUG_EX(EX_EXCEPTION_CAUGHT, e);
|
||||||
|
wslay_event_set_error(wsctx, WSLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
void addResponse(WebSocketSession* wsSession, const RpcResponse& res)
|
||||||
|
{
|
||||||
|
std::string response = toJson(res, "", false);
|
||||||
|
wsSession->addTextMessage(response);
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
void addResponse(WebSocketSession* wsSession,
|
||||||
|
const std::vector<RpcResponse>& results)
|
||||||
|
{
|
||||||
|
std::string response = toJsonBatch(results, "", false);
|
||||||
|
wsSession->addTextMessage(response);
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
void onMsgRecvCallback(wslay_event_context_ptr wsctx,
|
||||||
|
const struct wslay_event_on_msg_recv_arg* arg,
|
||||||
|
void* userData)
|
||||||
|
{
|
||||||
|
WebSocketSession* wsSession = reinterpret_cast<WebSocketSession*>(userData);
|
||||||
|
if(!wslay_is_ctrl_frame(arg->opcode)) {
|
||||||
|
// TODO Only process text frame
|
||||||
|
SharedHandle<ValueBase> json;
|
||||||
|
try {
|
||||||
|
json = json::decode(arg->msg, arg->msg_length);
|
||||||
|
} catch(RecoverableException& e) {
|
||||||
|
A2_LOG_INFO_EX("Failed to parse JSON-RPC request", e);
|
||||||
|
RpcResponse res
|
||||||
|
(createJsonRpcErrorResponse(-32700, "Parse error.", Null::g()));
|
||||||
|
addResponse(wsSession, res);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const Dict* jsondict = downcast<Dict>(json);
|
||||||
|
if(jsondict) {
|
||||||
|
RpcResponse res = processJsonRpcRequest(jsondict,
|
||||||
|
wsSession->getDownloadEngine());
|
||||||
|
addResponse(wsSession, res);
|
||||||
|
} else {
|
||||||
|
const List* jsonlist = downcast<List>(json);
|
||||||
|
if(jsonlist) {
|
||||||
|
// This is batch call
|
||||||
|
std::vector<RpcResponse> results;
|
||||||
|
for(List::ValueType::const_iterator i = jsonlist->begin(),
|
||||||
|
eoi = jsonlist->end(); i != eoi; ++i) {
|
||||||
|
const Dict* jsondict = downcast<Dict>(*i);
|
||||||
|
if(jsondict) {
|
||||||
|
RpcResponse r = processJsonRpcRequest
|
||||||
|
(jsondict, wsSession->getDownloadEngine());
|
||||||
|
results.push_back(r);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
addResponse(wsSession, results);
|
||||||
|
} else {
|
||||||
|
RpcResponse res(createJsonRpcErrorResponse
|
||||||
|
(-32600, "Invalid Request.", Null::g()));
|
||||||
|
addResponse(wsSession, res);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
RpcResponse res(createJsonRpcErrorResponse
|
||||||
|
(-32600, "Invalid Request.", Null::g()));
|
||||||
|
addResponse(wsSession, res);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
WebSocketSession::WebSocketSession(const SharedHandle<SocketCore>& socket,
|
||||||
|
DownloadEngine* e)
|
||||||
|
: socket_(socket),
|
||||||
|
e_(e)
|
||||||
|
{
|
||||||
|
wslay_event_callbacks callbacks;
|
||||||
|
memset(&callbacks, 0, sizeof(wslay_event_callbacks));
|
||||||
|
callbacks.recv_callback = recvCallback;
|
||||||
|
callbacks.send_callback = sendCallback;
|
||||||
|
callbacks.on_msg_recv_callback = onMsgRecvCallback;
|
||||||
|
int r = wslay_event_context_server_init(&wsctx_, &callbacks, this);
|
||||||
|
assert(r == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
WebSocketSession::~WebSocketSession()
|
||||||
|
{
|
||||||
|
wslay_event_context_free(wsctx_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WebSocketSession::wantRead()
|
||||||
|
{
|
||||||
|
return wslay_event_want_read(wsctx_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WebSocketSession::wantWrite()
|
||||||
|
{
|
||||||
|
return wslay_event_want_write(wsctx_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WebSocketSession::finish()
|
||||||
|
{
|
||||||
|
return !wantRead() && !wantWrite();
|
||||||
|
}
|
||||||
|
|
||||||
|
int WebSocketSession::onReadEvent()
|
||||||
|
{
|
||||||
|
if(wslay_event_recv(wsctx_) == 0) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int WebSocketSession::onWriteEvent()
|
||||||
|
{
|
||||||
|
if(wslay_event_send(wsctx_) == 0) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebSocketSession::addTextMessage(const std::string& msg)
|
||||||
|
{
|
||||||
|
// TODO Don't add text message if the size of outbound queue in
|
||||||
|
// wsctx_ exceeds certain limit.
|
||||||
|
wslay_event_msg arg = {
|
||||||
|
WSLAY_TEXT_FRAME, reinterpret_cast<const uint8_t*>(msg.c_str()), msg.size()
|
||||||
|
};
|
||||||
|
wslay_event_queue_msg(wsctx_, &arg);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WebSocketSession::closeReceived()
|
||||||
|
{
|
||||||
|
return wslay_event_get_close_received(wsctx_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WebSocketSession::closeSent()
|
||||||
|
{
|
||||||
|
return wslay_event_get_close_sent(wsctx_);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
} // namespace aria2
|
111
src/WebSocketSession.h
Normal file
111
src/WebSocketSession.h
Normal file
|
@ -0,0 +1,111 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#ifndef D_WEB_SOCKET_SESSION_H
|
||||||
|
#define D_WEB_SOCKET_SESSION_H
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
|
|
||||||
|
#include <wslay/wslay.h>
|
||||||
|
|
||||||
|
#include "SharedHandle.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
class SocketCore;
|
||||||
|
class DownloadEngine;
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
class WebSocketInteractionCommand;
|
||||||
|
|
||||||
|
class WebSocketSession {
|
||||||
|
public:
|
||||||
|
WebSocketSession(const SharedHandle<SocketCore>& socket,
|
||||||
|
DownloadEngine* e);
|
||||||
|
~WebSocketSession();
|
||||||
|
// Returns true if this session object wants to read data from the
|
||||||
|
// remote endpoint.
|
||||||
|
bool wantRead();
|
||||||
|
// Returns true if this session object wants to write data to the
|
||||||
|
// remote endpoint.
|
||||||
|
bool wantWrite();
|
||||||
|
// Returns true if the session ended and the underlying connection
|
||||||
|
// can be closed.
|
||||||
|
bool finish();
|
||||||
|
// Call this function when data is available to read. This function
|
||||||
|
// returns 0 if it succeeds, or -1.
|
||||||
|
int onReadEvent();
|
||||||
|
// Call this function when data can be sent without blocking. This
|
||||||
|
// function returns 0 if it succeeds, or -1.
|
||||||
|
int onWriteEvent();
|
||||||
|
// Adds text message |msg|. The message is queued and will be sent
|
||||||
|
// in onWriteEvent().
|
||||||
|
void addTextMessage(const std::string& msg);
|
||||||
|
// Returns true if the close frame is received.
|
||||||
|
bool closeReceived();
|
||||||
|
// Returns true if the close frame is sent.
|
||||||
|
bool closeSent();
|
||||||
|
|
||||||
|
const SharedHandle<SocketCore>& getSocket() const
|
||||||
|
{
|
||||||
|
return socket_;
|
||||||
|
}
|
||||||
|
|
||||||
|
DownloadEngine* getDownloadEngine()
|
||||||
|
{
|
||||||
|
return e_;
|
||||||
|
}
|
||||||
|
|
||||||
|
WebSocketInteractionCommand* getCommand()
|
||||||
|
{
|
||||||
|
return command_;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setCommand(WebSocketInteractionCommand* command)
|
||||||
|
{
|
||||||
|
command_ = command;
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
SharedHandle<SocketCore> socket_;
|
||||||
|
DownloadEngine* e_;
|
||||||
|
wslay_event_context_ptr wsctx_;
|
||||||
|
WebSocketInteractionCommand* command_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
} // namespace aria2
|
||||||
|
|
||||||
|
#endif // D_WEB_SOCKET_SESSION_H
|
83
src/WebSocketSessionMan.cc
Normal file
83
src/WebSocketSessionMan.cc
Normal file
|
@ -0,0 +1,83 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#include "WebSocketSessionMan.h"
|
||||||
|
#include "WebSocketSession.h"
|
||||||
|
#include "RequestGroup.h"
|
||||||
|
#include "json.h"
|
||||||
|
#include "util.h"
|
||||||
|
#include "WebSocketInteractionCommand.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
WebSocketSessionMan::WebSocketSessionMan() {}
|
||||||
|
|
||||||
|
WebSocketSessionMan::~WebSocketSessionMan() {}
|
||||||
|
|
||||||
|
void WebSocketSessionMan::addSession
|
||||||
|
(const SharedHandle<WebSocketSession>& wsSession)
|
||||||
|
{
|
||||||
|
sessions_.insert(wsSession);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebSocketSessionMan::removeSession
|
||||||
|
(const SharedHandle<WebSocketSession>& wsSession)
|
||||||
|
{
|
||||||
|
sessions_.erase(wsSession);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebSocketSessionMan::addNotification
|
||||||
|
(const std::string& method, const RequestGroup* group)
|
||||||
|
{
|
||||||
|
SharedHandle<Dict> dict = Dict::g();
|
||||||
|
dict->put("jsonrpc", "2.0");
|
||||||
|
dict->put("method", method);
|
||||||
|
SharedHandle<Dict> eventSpec = Dict::g();
|
||||||
|
eventSpec->put("gid", util::itos(group->getGID()));
|
||||||
|
SharedHandle<List> params = List::g();
|
||||||
|
params->append(eventSpec);
|
||||||
|
dict->put("params", params);
|
||||||
|
std::string msg = json::encode(dict);
|
||||||
|
for(WebSocketSessions::const_iterator i = sessions_.begin(),
|
||||||
|
eoi = sessions_.end(); i != eoi; ++i) {
|
||||||
|
(*i)->addTextMessage(msg);
|
||||||
|
(*i)->getCommand()->updateWriteCheck();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
} // namespace aria2
|
71
src/WebSocketSessionMan.h
Normal file
71
src/WebSocketSessionMan.h
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
/* <!-- copyright */
|
||||||
|
/*
|
||||||
|
* aria2 - The high speed download utility
|
||||||
|
*
|
||||||
|
* Copyright (C) 2012 Tatsuhiro Tsujikawa
|
||||||
|
*
|
||||||
|
* This program 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 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the copyright holders give
|
||||||
|
* permission to link the code of portions of this program with the
|
||||||
|
* OpenSSL library under certain conditions as described in each
|
||||||
|
* individual source file, and distribute linked combinations
|
||||||
|
* including the two.
|
||||||
|
* You must obey the GNU General Public License in all respects
|
||||||
|
* for all of the code used other than OpenSSL. If you modify
|
||||||
|
* file(s) with this exception, you may extend this exception to your
|
||||||
|
* version of the file(s), but you are not obligated to do so. If you
|
||||||
|
* do not wish to do so, delete this exception statement from your
|
||||||
|
* version. If you delete this exception statement from all source
|
||||||
|
* files in the program, then also delete it here.
|
||||||
|
*/
|
||||||
|
/* copyright --> */
|
||||||
|
#ifndef D_WEB_SOCKET_SESSION_MAN_H
|
||||||
|
#define D_WEB_SOCKET_SESSION_MAN_H
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
|
|
||||||
|
#include <set>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "SharedHandle.h"
|
||||||
|
#include "a2functional.h"
|
||||||
|
|
||||||
|
namespace aria2 {
|
||||||
|
|
||||||
|
class RequestGroup;
|
||||||
|
|
||||||
|
namespace rpc {
|
||||||
|
|
||||||
|
class WebSocketSession;
|
||||||
|
|
||||||
|
class WebSocketSessionMan {
|
||||||
|
public:
|
||||||
|
typedef std::set<SharedHandle<WebSocketSession>,
|
||||||
|
RefLess<WebSocketSession> > WebSocketSessions;
|
||||||
|
WebSocketSessionMan();
|
||||||
|
~WebSocketSessionMan();
|
||||||
|
void addSession(const SharedHandle<WebSocketSession>& wsSession);
|
||||||
|
void removeSession(const SharedHandle<WebSocketSession>& wsSession);
|
||||||
|
void addNotification(const std::string& method, const RequestGroup* group);
|
||||||
|
private:
|
||||||
|
WebSocketSessions sessions_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace rpc
|
||||||
|
|
||||||
|
} // aria2
|
||||||
|
|
||||||
|
#endif // D_WEB_SOCKET_SESSION_MAN_H
|
|
@ -331,6 +331,14 @@ struct DerefEqual<T> derefEqual(const T& t)
|
||||||
return DerefEqual<T>(t);
|
return DerefEqual<T>(t);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
struct RefLess {
|
||||||
|
bool operator()(const SharedHandle<T>& lhs, const SharedHandle<T>& rhs) const
|
||||||
|
{
|
||||||
|
return lhs.get() < rhs.get();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
||||||
#endif // D_A2_FUNCTIONAL_H
|
#endif // D_A2_FUNCTIONAL_H
|
||||||
|
|
127
src/json.cc
127
src/json.cc
|
@ -50,11 +50,9 @@ namespace json {
|
||||||
|
|
||||||
// Function prototype declaration
|
// Function prototype declaration
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decode
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decode(InputIterator first, InputIterator last, size_t depth);
|
||||||
std::string::const_iterator last,
|
|
||||||
size_t depth);
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
@ -64,9 +62,8 @@ const size_t MAX_STRUCTURE_DEPTH = 100;
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::string::const_iterator skipWs
|
template<typename InputIterator>
|
||||||
(std::string::const_iterator first,
|
InputIterator skipWs(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
while(first != last && std::find(vbegin(WS), vend(WS), *first) != vend(WS)) {
|
while(first != last && std::find(vbegin(WS), vend(WS), *first) != vend(WS)) {
|
||||||
++first;
|
++first;
|
||||||
|
@ -76,9 +73,8 @@ std::string::const_iterator skipWs
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
void checkEof
|
template<typename InputIterator>
|
||||||
(std::string::const_iterator first,
|
void checkEof(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
if(first == last) {
|
if(first == last) {
|
||||||
throw DL_ABORT_EX2("JSON decoding failed: unexpected EOF",
|
throw DL_ABORT_EX2("JSON decoding failed: unexpected EOF",
|
||||||
|
@ -88,11 +84,10 @@ void checkEof
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::string::const_iterator
|
template<typename InputIterator>
|
||||||
decodeKeyword
|
InputIterator
|
||||||
(std::string::const_iterator first,
|
decodeKeyword(InputIterator first, InputIterator last,
|
||||||
std::string::const_iterator last,
|
const std::string& keyword)
|
||||||
const std::string& keyword)
|
|
||||||
{
|
{
|
||||||
size_t len = keyword.size();
|
size_t len = keyword.size();
|
||||||
for(size_t i = 0; i < len; ++i) {
|
for(size_t i = 0; i < len; ++i) {
|
||||||
|
@ -109,10 +104,9 @@ decodeKeyword
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decodeTrue
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decodeTrue(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
first = decodeKeyword(first, last, "true");
|
first = decodeKeyword(first, last, "true");
|
||||||
return std::make_pair(Bool::gTrue(), first);
|
return std::make_pair(Bool::gTrue(), first);
|
||||||
|
@ -120,10 +114,9 @@ decodeTrue
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decodeFalse
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decodeFalse(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
first = decodeKeyword(first, last, "false");
|
first = decodeKeyword(first, last, "false");
|
||||||
return std::make_pair(Bool::gFalse(), first);
|
return std::make_pair(Bool::gFalse(), first);
|
||||||
|
@ -131,10 +124,9 @@ decodeFalse
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decodeNull
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decodeNull(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
first = decodeKeyword(first, last, "null");
|
first = decodeKeyword(first, last, "null");
|
||||||
return std::make_pair(Null::g(), first);
|
return std::make_pair(Null::g(), first);
|
||||||
|
@ -142,15 +134,14 @@ decodeNull
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decodeString
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decodeString(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
// Consume first char, assuming it is '"'.
|
// Consume first char, assuming it is '"'.
|
||||||
++first;
|
++first;
|
||||||
std::string s;
|
std::string s;
|
||||||
std::string::const_iterator offset = first;
|
InputIterator offset = first;
|
||||||
while(first != last) {
|
while(first != last) {
|
||||||
if(*first == '"') {
|
if(*first == '"') {
|
||||||
break;
|
break;
|
||||||
|
@ -161,7 +152,7 @@ decodeString
|
||||||
checkEof(first, last);
|
checkEof(first, last);
|
||||||
if(*first == 'u') {
|
if(*first == 'u') {
|
||||||
++first;
|
++first;
|
||||||
std::string::const_iterator uchars = first;
|
InputIterator uchars = first;
|
||||||
for(int i = 0; i < 4; ++i, ++first) {
|
for(int i = 0; i < 4; ++i, ++first) {
|
||||||
checkEof(first, last);
|
checkEof(first, last);
|
||||||
}
|
}
|
||||||
|
@ -182,7 +173,7 @@ decodeString
|
||||||
error_code::JSON_PARSE_ERROR);
|
error_code::JSON_PARSE_ERROR);
|
||||||
}
|
}
|
||||||
first += 2;
|
first += 2;
|
||||||
std::string::const_iterator uchars = first;
|
InputIterator uchars = first;
|
||||||
for(int i = 0; i < 4; ++i, ++first) {
|
for(int i = 0; i < 4; ++i, ++first) {
|
||||||
checkEof(first, last);
|
checkEof(first, last);
|
||||||
}
|
}
|
||||||
|
@ -242,9 +233,8 @@ decodeString
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
void checkEmptyDigit
|
template<typename InputIterator>
|
||||||
(std::string::const_iterator first,
|
void checkEmptyDigit(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
if(std::distance(first, last) == 0) {
|
if(std::distance(first, last) == 0) {
|
||||||
throw DL_ABORT_EX2("JSON decoding failed: zero DIGIT.",
|
throw DL_ABORT_EX2("JSON decoding failed: zero DIGIT.",
|
||||||
|
@ -254,9 +244,8 @@ void checkEmptyDigit
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
void checkLeadingZero
|
template<typename InputIterator>
|
||||||
(std::string::const_iterator first,
|
void checkLeadingZero(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
if(std::distance(first, last) > 2 && *first == '0') {
|
if(std::distance(first, last) > 2 && *first == '0') {
|
||||||
throw DL_ABORT_EX2("JSON decoding failed: leading zero.",
|
throw DL_ABORT_EX2("JSON decoding failed: leading zero.",
|
||||||
|
@ -266,17 +255,16 @@ void checkLeadingZero
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decodeNumber
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decodeNumber(InputIterator first, InputIterator last)
|
||||||
std::string::const_iterator last)
|
|
||||||
{
|
{
|
||||||
std::string s;
|
std::string s;
|
||||||
if(*first == '-') {
|
if(*first == '-') {
|
||||||
s.append(first, first+1);
|
s.append(first, first+1);
|
||||||
++first;
|
++first;
|
||||||
}
|
}
|
||||||
std::string::const_iterator offset = first;
|
InputIterator offset = first;
|
||||||
while(first != last && in(*first, '0', '9')) {
|
while(first != last && in(*first, '0', '9')) {
|
||||||
++first;
|
++first;
|
||||||
}
|
}
|
||||||
|
@ -336,11 +324,9 @@ void checkDepth(size_t depth)
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decodeArray
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decodeArray(InputIterator first, InputIterator last, size_t depth)
|
||||||
std::string::const_iterator last,
|
|
||||||
size_t depth)
|
|
||||||
{
|
{
|
||||||
checkDepth(depth);
|
checkDepth(depth);
|
||||||
SharedHandle<List> list = List::g();
|
SharedHandle<List> list = List::g();
|
||||||
|
@ -350,7 +336,7 @@ decodeArray
|
||||||
checkEof(first, last);
|
checkEof(first, last);
|
||||||
if(*first != ']') {
|
if(*first != ']') {
|
||||||
while(1) {
|
while(1) {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
r = decode(first, last, depth);
|
r = decode(first, last, depth);
|
||||||
list->append(r.first);
|
list->append(r.first);
|
||||||
first = r.second;
|
first = r.second;
|
||||||
|
@ -372,11 +358,9 @@ decodeArray
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decodeObject
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decodeObject(InputIterator first, InputIterator last, size_t depth)
|
||||||
std::string::const_iterator last,
|
|
||||||
size_t depth)
|
|
||||||
{
|
{
|
||||||
checkDepth(depth);
|
checkDepth(depth);
|
||||||
SharedHandle<Dict> dict = Dict::g();
|
SharedHandle<Dict> dict = Dict::g();
|
||||||
|
@ -386,7 +370,7 @@ decodeObject
|
||||||
checkEof(first, last);
|
checkEof(first, last);
|
||||||
if(*first != '}') {
|
if(*first != '}') {
|
||||||
while(1) {
|
while(1) {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
keyRet = decodeString(first, last);
|
keyRet = decodeString(first, last);
|
||||||
first = keyRet.second;
|
first = keyRet.second;
|
||||||
first = skipWs(first, last);
|
first = skipWs(first, last);
|
||||||
|
@ -396,7 +380,7 @@ decodeObject
|
||||||
error_code::JSON_PARSE_ERROR);
|
error_code::JSON_PARSE_ERROR);
|
||||||
}
|
}
|
||||||
++first;
|
++first;
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
valueRet = decode(first, last, depth);
|
valueRet = decode(first, last, depth);
|
||||||
dict->put(downcast<String>(keyRet.first)->s(), valueRet.first);
|
dict->put(downcast<String>(keyRet.first)->s(), valueRet.first);
|
||||||
first = valueRet.second;
|
first = valueRet.second;
|
||||||
|
@ -420,11 +404,9 @@ decodeObject
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator>
|
template<typename InputIterator>
|
||||||
decode
|
std::pair<SharedHandle<ValueBase>, InputIterator>
|
||||||
(std::string::const_iterator first,
|
decode(InputIterator first, InputIterator last, size_t depth)
|
||||||
std::string::const_iterator last,
|
|
||||||
size_t depth)
|
|
||||||
{
|
{
|
||||||
first = skipWs(first, last);
|
first = skipWs(first, last);
|
||||||
if(first == last) {
|
if(first == last) {
|
||||||
|
@ -454,17 +436,16 @@ decode
|
||||||
}
|
}
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
SharedHandle<ValueBase> decode(const std::string& json)
|
template<typename InputIterator>
|
||||||
|
SharedHandle<ValueBase> decode(InputIterator first, InputIterator last)
|
||||||
{
|
{
|
||||||
std::string::const_iterator first = json.begin();
|
|
||||||
std::string::const_iterator last = json.end();
|
|
||||||
first = skipWs(first, last);
|
first = skipWs(first, last);
|
||||||
if(first == last) {
|
if(first == last) {
|
||||||
throw DL_ABORT_EX2("JSON decoding failed:"
|
throw DL_ABORT_EX2("JSON decoding failed:"
|
||||||
" Unexpected EOF in term context.",
|
" Unexpected EOF in term context.",
|
||||||
error_code::JSON_PARSE_ERROR);
|
error_code::JSON_PARSE_ERROR);
|
||||||
}
|
}
|
||||||
std::pair<SharedHandle<ValueBase>, std::string::const_iterator> r;
|
std::pair<SharedHandle<ValueBase>, InputIterator> r;
|
||||||
if(*first == '[') {
|
if(*first == '[') {
|
||||||
r = decodeArray(first, last, 1);
|
r = decodeArray(first, last, 1);
|
||||||
} else if(*first == '{') {
|
} else if(*first == '{') {
|
||||||
|
@ -477,6 +458,16 @@ SharedHandle<ValueBase> decode(const std::string& json)
|
||||||
return r.first;
|
return r.first;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SharedHandle<ValueBase> decode(const std::string& json)
|
||||||
|
{
|
||||||
|
return decode(json.begin(), json.end());
|
||||||
|
}
|
||||||
|
|
||||||
|
SharedHandle<ValueBase> decode(const unsigned char* json, size_t len)
|
||||||
|
{
|
||||||
|
return decode(json, json+len);
|
||||||
|
}
|
||||||
|
|
||||||
std::string jsonEscape(const std::string& s)
|
std::string jsonEscape(const std::string& s)
|
||||||
{
|
{
|
||||||
std::string t;
|
std::string t;
|
||||||
|
|
|
@ -45,6 +45,8 @@ namespace json {
|
||||||
// Parses JSON text defined in RFC4627.
|
// Parses JSON text defined in RFC4627.
|
||||||
SharedHandle<ValueBase> decode(const std::string& json);
|
SharedHandle<ValueBase> decode(const std::string& json);
|
||||||
|
|
||||||
|
SharedHandle<ValueBase> decode(const unsigned char* json, size_t len);
|
||||||
|
|
||||||
std::string jsonEscape(const std::string& s);
|
std::string jsonEscape(const std::string& s);
|
||||||
|
|
||||||
template<typename OutputStream>
|
template<typename OutputStream>
|
||||||
|
|
|
@ -38,6 +38,12 @@
|
||||||
#include "XmlRpcRequestParserStateMachine.h"
|
#include "XmlRpcRequestParserStateMachine.h"
|
||||||
#include "message.h"
|
#include "message.h"
|
||||||
#include "DlAbortEx.h"
|
#include "DlAbortEx.h"
|
||||||
|
#include "DownloadEngine.h"
|
||||||
|
#include "RpcMethod.h"
|
||||||
|
#include "RpcResponse.h"
|
||||||
|
#include "RpcMethodFactory.h"
|
||||||
|
#include "LogFactory.h"
|
||||||
|
#include "fmt.h"
|
||||||
|
|
||||||
namespace aria2 {
|
namespace aria2 {
|
||||||
|
|
||||||
|
@ -60,6 +66,51 @@ RpcRequest xmlParseMemory(const char* xml, size_t size)
|
||||||
}
|
}
|
||||||
#endif // ENABLE_XML_RPC
|
#endif // ENABLE_XML_RPC
|
||||||
|
|
||||||
|
RpcResponse createJsonRpcErrorResponse(int code,
|
||||||
|
const std::string& msg,
|
||||||
|
const SharedHandle<ValueBase>& id)
|
||||||
|
{
|
||||||
|
SharedHandle<Dict> params = Dict::g();
|
||||||
|
params->put("code", Integer::g(code));
|
||||||
|
params->put("message", msg);
|
||||||
|
rpc::RpcResponse res(code, params, id);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
RpcResponse processJsonRpcRequest(const Dict* jsondict, DownloadEngine* e)
|
||||||
|
{
|
||||||
|
SharedHandle<ValueBase> id = jsondict->get("id");
|
||||||
|
if(!id) {
|
||||||
|
return createJsonRpcErrorResponse(-32600, "Invalid Request.", Null::g());
|
||||||
|
}
|
||||||
|
const String* methodName = downcast<String>(jsondict->get("method"));
|
||||||
|
if(!methodName) {
|
||||||
|
return createJsonRpcErrorResponse(-32600, "Invalid Request.", id);
|
||||||
|
}
|
||||||
|
SharedHandle<List> params;
|
||||||
|
const SharedHandle<ValueBase>& tempParams = jsondict->get("params");
|
||||||
|
if(downcast<List>(tempParams)) {
|
||||||
|
params = static_pointer_cast<List>(tempParams);
|
||||||
|
} else if(!tempParams) {
|
||||||
|
params = List::g();
|
||||||
|
} else {
|
||||||
|
// TODO No support for Named params
|
||||||
|
return createJsonRpcErrorResponse(-32602, "Invalid params.", id);
|
||||||
|
}
|
||||||
|
rpc::RpcRequest req(methodName->s(), params, id);
|
||||||
|
req.jsonRpc = true;
|
||||||
|
SharedHandle<rpc::RpcMethod> method;
|
||||||
|
try {
|
||||||
|
method = rpc::RpcMethodFactory::create(req.methodName);
|
||||||
|
} catch(RecoverableException& e) {
|
||||||
|
A2_LOG_INFO_EX(EX_EXCEPTION_CAUGHT, e);
|
||||||
|
return createJsonRpcErrorResponse(-32601, "Method not found.", id);
|
||||||
|
}
|
||||||
|
A2_LOG_INFO(fmt("Executing RPC method %s", req.methodName.c_str()));
|
||||||
|
rpc::RpcResponse res = method->execute(req, e);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace rpc
|
} // namespace rpc
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
|
@ -38,17 +38,34 @@
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "SharedHandle.h"
|
||||||
|
|
||||||
namespace aria2 {
|
namespace aria2 {
|
||||||
|
|
||||||
|
class ValueBase;
|
||||||
|
class Dict;
|
||||||
|
class DownloadEngine;
|
||||||
|
|
||||||
namespace rpc {
|
namespace rpc {
|
||||||
|
|
||||||
struct RpcRequest;
|
struct RpcRequest;
|
||||||
|
struct RpcResponse;
|
||||||
|
|
||||||
#ifdef ENABLE_XML_RPC
|
#ifdef ENABLE_XML_RPC
|
||||||
RpcRequest xmlParseMemory(const char* xml, size_t size);
|
RpcRequest xmlParseMemory(const char* xml, size_t size);
|
||||||
#endif // ENABLE_XML_RPC
|
#endif // ENABLE_XML_RPC
|
||||||
|
|
||||||
|
// Creates error response. The |code| is the JSON-RPC error code. The
|
||||||
|
// |msg| is the error message. The |id| is the id of the request .
|
||||||
|
RpcResponse createJsonRpcErrorResponse(int code,
|
||||||
|
const std::string& msg,
|
||||||
|
const SharedHandle<ValueBase>& id);
|
||||||
|
|
||||||
|
// Processes JSON-RPC request |jsondict| and returns the result.
|
||||||
|
RpcResponse processJsonRpcRequest(const Dict* jsondict, DownloadEngine* e);
|
||||||
|
|
||||||
} // namespace rpc
|
} // namespace rpc
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
#include "SingletonHolder.h"
|
#include "SingletonHolder.h"
|
||||||
#include "SharedHandle.h"
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
#include <cppunit/extensions/HelperMacros.h>
|
#include <cppunit/extensions/HelperMacros.h>
|
||||||
|
|
||||||
|
#include "SharedHandle.h"
|
||||||
|
|
||||||
namespace aria2 {
|
namespace aria2 {
|
||||||
|
|
||||||
class SingletonHolderTest : public CppUnit::TestFixture {
|
class SingletonHolderTest : public CppUnit::TestFixture {
|
||||||
|
@ -35,26 +38,16 @@ public:
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef SharedHandle<M> MHandle;
|
|
||||||
typedef SharedHandle<int> IntHandle;
|
|
||||||
|
|
||||||
void SingletonHolderTest::testInstance()
|
void SingletonHolderTest::testInstance()
|
||||||
{
|
{
|
||||||
MHandle m(new M("Hello world."));
|
M m("Hello world.");
|
||||||
SingletonHolder<MHandle>::instance(m);
|
SingletonHolder<M>::instance(&m);
|
||||||
|
CPPUNIT_ASSERT_EQUAL(std::string("Hello world."),
|
||||||
std::cerr << SingletonHolder<MHandle>::instance()->greeting() << std::endl;
|
SingletonHolder<M>::instance()->greeting());
|
||||||
|
|
||||||
SingletonHolder<MHandle>::instance()->greeting("Yes, it worked!");
|
|
||||||
|
|
||||||
std::cerr << SingletonHolder<MHandle>::instance()->greeting() << std::endl;
|
|
||||||
|
|
||||||
IntHandle i(new int(100));
|
|
||||||
SingletonHolder<IntHandle>::instance(i);
|
|
||||||
std::cerr << *SingletonHolder<IntHandle>::instance() << std::endl;
|
|
||||||
|
|
||||||
std::cerr << SingletonHolder<MHandle>::instance()->greeting() << std::endl;
|
|
||||||
|
|
||||||
|
SingletonHolder<M>::instance()->greeting("Yes, it worked!");
|
||||||
|
CPPUNIT_ASSERT_EQUAL(std::string("Yes, it worked!"),
|
||||||
|
SingletonHolder<M>::instance()->greeting());
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue