|
Packit |
f0b94e |
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
Packit |
f0b94e |
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
Packit |
f0b94e |
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
Packit |
f0b94e |
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
#include "mozilla/net/NeckoChild.h"
|
|
Packit |
f0b94e |
#include "mozilla/net/FTPChannelChild.h"
|
|
Packit |
f0b94e |
using namespace mozilla;
|
|
Packit |
f0b94e |
using namespace mozilla::net;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
#include "nsFtpProtocolHandler.h"
|
|
Packit |
f0b94e |
#include "nsFTPChannel.h"
|
|
Packit |
f0b94e |
#include "nsIStandardURL.h"
|
|
Packit |
f0b94e |
#include "mozilla/Logging.h"
|
|
Packit |
f0b94e |
#include "nsIPrefService.h"
|
|
Packit |
f0b94e |
#include "nsIPrefBranch.h"
|
|
Packit |
f0b94e |
#include "nsIObserverService.h"
|
|
Packit |
f0b94e |
#include "nsEscape.h"
|
|
Packit |
f0b94e |
#include "nsAlgorithm.h"
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
//-----------------------------------------------------------------------------
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
//
|
|
Packit |
f0b94e |
// Log module for FTP Protocol logging...
|
|
Packit |
f0b94e |
//
|
|
Packit |
f0b94e |
// To enable logging (see prlog.h for full details):
|
|
Packit |
f0b94e |
//
|
|
Packit |
f0b94e |
// set MOZ_LOG=nsFtp:5
|
|
Packit |
f0b94e |
// set MOZ_LOG_FILE=ftp.log
|
|
Packit |
f0b94e |
//
|
|
Packit |
f0b94e |
// This enables LogLevel::Debug level information and places all output in
|
|
Packit |
f0b94e |
// the file ftp.log.
|
|
Packit |
f0b94e |
//
|
|
Packit |
f0b94e |
LazyLogModule gFTPLog("nsFtp");
|
|
Packit |
f0b94e |
#undef LOG
|
|
Packit |
f0b94e |
#define LOG(args) MOZ_LOG(gFTPLog, mozilla::LogLevel::Debug, args)
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
//-----------------------------------------------------------------------------
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
#define IDLE_TIMEOUT_PREF "network.ftp.idleConnectionTimeout"
|
|
Packit |
f0b94e |
#define IDLE_CONNECTION_LIMIT 8 /* TODO pref me */
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
#define ENABLED_PREF "network.ftp.enabled"
|
|
Packit |
f0b94e |
#define QOS_DATA_PREF "network.ftp.data.qos"
|
|
Packit |
f0b94e |
#define QOS_CONTROL_PREF "network.ftp.control.qos"
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsFtpProtocolHandler *gFtpHandler = nullptr;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
//-----------------------------------------------------------------------------
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::nsFtpProtocolHandler()
|
|
Packit |
f0b94e |
: mIdleTimeout(-1),
|
|
Packit |
f0b94e |
mEnabled(true),
|
|
Packit |
f0b94e |
mSessionId(0),
|
|
Packit |
f0b94e |
mControlQoSBits(0x00),
|
|
Packit |
f0b94e |
mDataQoSBits(0x00) {
|
|
Packit |
f0b94e |
LOG(("FTP:creating handler @%p\n", this));
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
gFtpHandler = this;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::~nsFtpProtocolHandler() {
|
|
Packit |
f0b94e |
LOG(("FTP:destroying handler @%p\n", this));
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_ASSERTION(mRootConnectionList.Length() == 0, "why wasn't Observe called?");
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
gFtpHandler = nullptr;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMPL_ISUPPORTS(nsFtpProtocolHandler, nsIProtocolHandler,
|
|
Packit |
f0b94e |
nsIProxiedProtocolHandler, nsIObserver,
|
|
Packit |
f0b94e |
nsISupportsWeakReference)
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsresult nsFtpProtocolHandler::Init() {
|
|
Packit |
f0b94e |
if (IsNeckoChild()) NeckoChild::InitNeckoChild();
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
if (mIdleTimeout == -1) {
|
|
Packit |
f0b94e |
nsresult rv;
|
|
Packit |
f0b94e |
nsCOMPtr<nsIPrefBranch> branch =
|
|
Packit |
f0b94e |
do_GetService(NS_PREFSERVICE_CONTRACTID, &rv;;
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) return rv;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->GetIntPref(IDLE_TIMEOUT_PREF, &mIdleTimeout);
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) mIdleTimeout = 5 * 60; // 5 minute default
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->AddObserver(IDLE_TIMEOUT_PREF, this, true);
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) return rv;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->GetBoolPref(ENABLED_PREF, &mEnabled);
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) mEnabled = true;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->AddObserver(ENABLED_PREF, this, true);
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) return rv;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
int32_t val;
|
|
Packit |
f0b94e |
rv = branch->GetIntPref(QOS_DATA_PREF, &val;;
|
|
Packit |
f0b94e |
if (NS_SUCCEEDED(rv)) mDataQoSBits = (uint8_t)clamped(val, 0, 0xff);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->AddObserver(QOS_DATA_PREF, this, true);
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) return rv;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->GetIntPref(QOS_CONTROL_PREF, &val;;
|
|
Packit |
f0b94e |
if (NS_SUCCEEDED(rv)) mControlQoSBits = (uint8_t)clamped(val, 0, 0xff);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->AddObserver(QOS_CONTROL_PREF, this, true);
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) return rv;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsCOMPtr<nsIObserverService> observerService =
|
|
Packit |
f0b94e |
mozilla::services::GetObserverService();
|
|
Packit |
f0b94e |
if (observerService) {
|
|
Packit |
f0b94e |
observerService->AddObserver(this, "network:offline-about-to-go-offline",
|
|
Packit |
f0b94e |
true);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
observerService->AddObserver(this, "net:clear-active-logins", true);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
return NS_OK;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
//-----------------------------------------------------------------------------
|
|
Packit |
f0b94e |
// nsIProtocolHandler methods:
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::GetScheme(nsACString &result) {
|
|
Packit |
f0b94e |
result.AssignLiteral("ftp");
|
|
Packit |
f0b94e |
return NS_OK;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::GetDefaultPort(int32_t *result) {
|
|
Packit |
f0b94e |
*result = 21;
|
|
Packit |
f0b94e |
return NS_OK;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::GetProtocolFlags(uint32_t *result) {
|
|
Packit |
f0b94e |
*result = URI_STD | ALLOWS_PROXY | ALLOWS_PROXY_HTTP | URI_LOADABLE_BY_ANYONE;
|
|
Packit |
f0b94e |
return NS_OK;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::NewURI(const nsACString &aSpec, const char *aCharset,
|
|
Packit |
f0b94e |
nsIURI *aBaseURI, nsIURI **result) {
|
|
Packit |
f0b94e |
if (!mEnabled) {
|
|
Packit |
f0b94e |
return NS_ERROR_UNKNOWN_PROTOCOL;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
nsAutoCString spec(aSpec);
|
|
Packit |
f0b94e |
spec.Trim(" \t\n\r"); // Match NS_IsAsciiWhitespace instead of HTML5
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
char *fwdPtr = spec.BeginWriting();
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
// now unescape it... %xx reduced inline to resulting character
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
int32_t len = NS_UnescapeURL(fwdPtr);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
// NS_UnescapeURL() modified spec's buffer, truncate to ensure
|
|
Packit |
f0b94e |
// spec knows its new length.
|
|
Packit |
f0b94e |
spec.Truncate(len);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
// return an error if we find a NUL, CR, or LF in the path
|
|
Packit |
f0b94e |
if (spec.FindCharInSet(CRLF) >= 0 || spec.FindChar('\0') >= 0)
|
|
Packit |
f0b94e |
return NS_ERROR_MALFORMED_URI;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsCOMPtr<nsIURI> base(aBaseURI);
|
|
Packit |
f0b94e |
return NS_MutateURI(NS_STANDARDURLMUTATOR_CONTRACTID)
|
|
Packit |
f0b94e |
.Apply(NS_MutatorMethod(&nsIStandardURLMutator::Init,
|
|
Packit |
f0b94e |
nsIStandardURL::URLTYPE_AUTHORITY, 21,
|
|
Packit |
f0b94e |
nsCString(aSpec), aCharset, base, nullptr))
|
|
Packit |
f0b94e |
.Finalize(result);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::NewChannel2(nsIURI *url, nsILoadInfo *aLoadInfo,
|
|
Packit |
f0b94e |
nsIChannel **result) {
|
|
Packit |
f0b94e |
return NewProxiedChannel2(url, nullptr, 0, nullptr, aLoadInfo, result);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::NewChannel(nsIURI *url, nsIChannel **result) {
|
|
Packit |
f0b94e |
return NewChannel2(url, nullptr, result);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::NewProxiedChannel2(nsIURI *uri, nsIProxyInfo *proxyInfo,
|
|
Packit |
f0b94e |
uint32_t proxyResolveFlags,
|
|
Packit |
f0b94e |
nsIURI *proxyURI,
|
|
Packit |
f0b94e |
nsILoadInfo *aLoadInfo,
|
|
Packit |
f0b94e |
nsIChannel **result) {
|
|
Packit |
f0b94e |
NS_ENSURE_ARG_POINTER(uri);
|
|
Packit |
f0b94e |
RefPtr<nsBaseChannel> channel;
|
|
Packit |
f0b94e |
if (IsNeckoChild())
|
|
Packit |
f0b94e |
channel = new FTPChannelChild(uri);
|
|
Packit |
f0b94e |
else
|
|
Packit |
f0b94e |
channel = new nsFtpChannel(uri, proxyInfo);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsresult rv = channel->Init();
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) {
|
|
Packit |
f0b94e |
return rv;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
// set the loadInfo on the new channel
|
|
Packit |
f0b94e |
rv = channel->SetLoadInfo(aLoadInfo);
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) {
|
|
Packit |
f0b94e |
return rv;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
channel.forget(result);
|
|
Packit |
f0b94e |
return rv;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::NewProxiedChannel(nsIURI *uri, nsIProxyInfo *proxyInfo,
|
|
Packit |
f0b94e |
uint32_t proxyResolveFlags,
|
|
Packit |
f0b94e |
nsIURI *proxyURI, nsIChannel **result) {
|
|
Packit |
f0b94e |
return NewProxiedChannel2(uri, proxyInfo, proxyResolveFlags, proxyURI,
|
|
Packit |
f0b94e |
nullptr /*loadinfo*/, result);
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::AllowPort(int32_t port, const char *scheme,
|
|
Packit |
f0b94e |
bool *_retval) {
|
|
Packit |
f0b94e |
*_retval = (port == 21 || port == 22);
|
|
Packit |
f0b94e |
return NS_OK;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
// connection cache methods
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
void nsFtpProtocolHandler::Timeout(nsITimer *aTimer, void *aClosure) {
|
|
Packit |
f0b94e |
LOG(("FTP:timeout reached for %p\n", aClosure));
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
bool found = gFtpHandler->mRootConnectionList.RemoveElement(aClosure);
|
|
Packit |
f0b94e |
if (!found) {
|
|
Packit |
f0b94e |
NS_ERROR("timerStruct not found");
|
|
Packit |
f0b94e |
return;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
timerStruct *s = (timerStruct *)aClosure;
|
|
Packit |
f0b94e |
delete s;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsresult nsFtpProtocolHandler::RemoveConnection(
|
|
Packit |
f0b94e |
nsIURI *aKey, nsFtpControlConnection **_retval) {
|
|
Packit |
f0b94e |
NS_ASSERTION(_retval, "null pointer");
|
|
Packit |
f0b94e |
NS_ASSERTION(aKey, "null pointer");
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
*_retval = nullptr;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsAutoCString spec;
|
|
Packit |
f0b94e |
aKey->GetPrePath(spec);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
LOG(("FTP:removing connection for %s\n", spec.get()));
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
timerStruct *ts = nullptr;
|
|
Packit |
f0b94e |
uint32_t i;
|
|
Packit |
f0b94e |
bool found = false;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
for (i = 0; i < mRootConnectionList.Length(); ++i) {
|
|
Packit |
f0b94e |
ts = mRootConnectionList[i];
|
|
Packit |
f0b94e |
if (strcmp(spec.get(), ts->key) == 0) {
|
|
Packit |
f0b94e |
found = true;
|
|
Packit |
f0b94e |
mRootConnectionList.RemoveElementAt(i);
|
|
Packit |
f0b94e |
break;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
if (!found) return NS_ERROR_FAILURE;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
// swap connection ownership
|
|
Packit |
f0b94e |
ts->conn.forget(_retval);
|
|
Packit |
f0b94e |
delete ts;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
return NS_OK;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsresult nsFtpProtocolHandler::InsertConnection(nsIURI *aKey,
|
|
Packit |
f0b94e |
nsFtpControlConnection *aConn) {
|
|
Packit |
f0b94e |
NS_ASSERTION(aConn, "null pointer");
|
|
Packit |
f0b94e |
NS_ASSERTION(aKey, "null pointer");
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
if (aConn->mSessionId != mSessionId) return NS_ERROR_FAILURE;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsAutoCString spec;
|
|
Packit |
f0b94e |
aKey->GetPrePath(spec);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
LOG(("FTP:inserting connection for %s\n", spec.get()));
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
timerStruct *ts = new timerStruct();
|
|
Packit |
f0b94e |
if (!ts) return NS_ERROR_OUT_OF_MEMORY;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
nsCOMPtr<nsITimer> timer;
|
|
Packit |
f0b94e |
nsresult rv = NS_NewTimerWithFuncCallback(
|
|
Packit |
f0b94e |
getter_AddRefs(timer), nsFtpProtocolHandler::Timeout, ts,
|
|
Packit |
f0b94e |
mIdleTimeout * 1000, nsITimer::TYPE_REPEATING_SLACK,
|
|
Packit |
f0b94e |
"nsFtpProtocolHandler::InsertConnection");
|
|
Packit |
f0b94e |
if (NS_FAILED(rv)) {
|
|
Packit |
f0b94e |
delete ts;
|
|
Packit |
f0b94e |
return rv;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
ts->key = ToNewCString(spec);
|
|
Packit |
f0b94e |
if (!ts->key) {
|
|
Packit |
f0b94e |
delete ts;
|
|
Packit |
f0b94e |
return NS_ERROR_OUT_OF_MEMORY;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
// ts->conn is a RefPtr
|
|
Packit |
f0b94e |
ts->conn = aConn;
|
|
Packit |
f0b94e |
ts->timer = timer;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
//
|
|
Packit |
f0b94e |
// limit number of idle connections. if limit is reached, then prune
|
|
Packit |
f0b94e |
// eldest connection with matching key. if none matching, then prune
|
|
Packit |
f0b94e |
// eldest connection.
|
|
Packit |
f0b94e |
//
|
|
Packit |
f0b94e |
if (mRootConnectionList.Length() == IDLE_CONNECTION_LIMIT) {
|
|
Packit |
f0b94e |
uint32_t i;
|
|
Packit |
f0b94e |
for (i = 0; i < mRootConnectionList.Length(); ++i) {
|
|
Packit |
f0b94e |
timerStruct *candidate = mRootConnectionList[i];
|
|
Packit |
f0b94e |
if (strcmp(candidate->key, ts->key) == 0) {
|
|
Packit |
f0b94e |
mRootConnectionList.RemoveElementAt(i);
|
|
Packit |
f0b94e |
delete candidate;
|
|
Packit |
f0b94e |
break;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
if (mRootConnectionList.Length() == IDLE_CONNECTION_LIMIT) {
|
|
Packit |
f0b94e |
timerStruct *eldest = mRootConnectionList[0];
|
|
Packit |
f0b94e |
mRootConnectionList.RemoveElementAt(0);
|
|
Packit |
f0b94e |
delete eldest;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
mRootConnectionList.AppendElement(ts);
|
|
Packit |
f0b94e |
return NS_OK;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
//-----------------------------------------------------------------------------
|
|
Packit |
f0b94e |
// nsIObserver
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
NS_IMETHODIMP
|
|
Packit |
f0b94e |
nsFtpProtocolHandler::Observe(nsISupports *aSubject, const char *aTopic,
|
|
Packit |
f0b94e |
const char16_t *aData) {
|
|
Packit |
f0b94e |
LOG(("FTP:observing [%s]\n", aTopic));
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
if (!strcmp(aTopic, NS_PREFBRANCH_PREFCHANGE_TOPIC_ID)) {
|
|
Packit |
f0b94e |
nsCOMPtr<nsIPrefBranch> branch = do_QueryInterface(aSubject);
|
|
Packit |
f0b94e |
if (!branch) {
|
|
Packit |
f0b94e |
NS_ERROR("no prefbranch");
|
|
Packit |
f0b94e |
return NS_ERROR_UNEXPECTED;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
int32_t val;
|
|
Packit |
f0b94e |
nsresult rv = branch->GetIntPref(IDLE_TIMEOUT_PREF, &val;;
|
|
Packit |
f0b94e |
if (NS_SUCCEEDED(rv)) mIdleTimeout = val;
|
|
Packit |
f0b94e |
bool enabled;
|
|
Packit |
f0b94e |
rv = branch->GetBoolPref(ENABLED_PREF, &enabled);
|
|
Packit |
f0b94e |
if (NS_SUCCEEDED(rv)) mEnabled = enabled;
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->GetIntPref(QOS_DATA_PREF, &val;;
|
|
Packit |
f0b94e |
if (NS_SUCCEEDED(rv)) mDataQoSBits = (uint8_t)clamped(val, 0, 0xff);
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
rv = branch->GetIntPref(QOS_CONTROL_PREF, &val;;
|
|
Packit |
f0b94e |
if (NS_SUCCEEDED(rv)) mControlQoSBits = (uint8_t)clamped(val, 0, 0xff);
|
|
Packit |
f0b94e |
} else if (!strcmp(aTopic, "network:offline-about-to-go-offline")) {
|
|
Packit |
f0b94e |
ClearAllConnections();
|
|
Packit |
f0b94e |
} else if (!strcmp(aTopic, "net:clear-active-logins")) {
|
|
Packit |
f0b94e |
ClearAllConnections();
|
|
Packit |
f0b94e |
mSessionId++;
|
|
Packit |
f0b94e |
} else {
|
|
Packit |
f0b94e |
NS_NOTREACHED("unexpected topic");
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
return NS_OK;
|
|
Packit |
f0b94e |
}
|
|
Packit |
f0b94e |
|
|
Packit |
f0b94e |
void nsFtpProtocolHandler::ClearAllConnections() {
|
|
Packit |
f0b94e |
uint32_t i;
|
|
Packit |
f0b94e |
for (i = 0; i < mRootConnectionList.Length(); ++i)
|
|
Packit |
f0b94e |
delete mRootConnectionList[i];
|
|
Packit |
f0b94e |
mRootConnectionList.Clear();
|
|
Packit |
f0b94e |
}
|