Blame netwerk/protocol/ftp/nsFtpProtocolHandler.cpp

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
}