blob: 139083d1932115b7f3088e44f904dc1780142b1d [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "net/http/http_stream_factory.h"
#include "base/logging.h"
#include "base/string_number_conversions.h"
#include "base/string_split.h"
#include "googleurl/src/gurl.h"
#include "net/base/host_mapping_rules.h"
#include "net/base/host_port_pair.h"
namespace net {
// WARNING: If you modify or add any static flags, you must keep them in sync
// with |ResetStaticSettingsToInit|. This is critical for unit test isolation.
// static
std::vector<std::string>* HttpStreamFactory::next_protos_ = NULL;
// static
bool HttpStreamFactory::enabled_protocols_[NUM_ALTERNATE_PROTOCOLS];
// static
bool HttpStreamFactory::spdy_enabled_ = true;
// static
bool HttpStreamFactory::use_alternate_protocols_ = false;
// static
bool HttpStreamFactory::force_spdy_over_ssl_ = true;
// static
bool HttpStreamFactory::force_spdy_always_ = false;
// static
std::list<HostPortPair>* HttpStreamFactory::forced_spdy_exclusions_ = NULL;
HttpStreamFactory::~HttpStreamFactory() {}
// static
void HttpStreamFactory::ResetStaticSettingsToInit() {
// WARNING: These must match the initializers above.
delete next_protos_;
delete forced_spdy_exclusions_;
next_protos_ = NULL;
spdy_enabled_ = true;
use_alternate_protocols_ = false;
force_spdy_over_ssl_ = true;
force_spdy_always_ = false;
forced_spdy_exclusions_ = NULL;
for (int i = 0; i < NUM_ALTERNATE_PROTOCOLS; ++i)
enabled_protocols_[i] = false;
}
void HttpStreamFactory::ProcessAlternateProtocol(
HttpServerProperties* http_server_properties,
const std::string& alternate_protocol_str,
const HostPortPair& http_host_port_pair) {
std::vector<std::string> port_protocol_vector;
base::SplitString(alternate_protocol_str, ':', &port_protocol_vector);
if (port_protocol_vector.size() != 2) {
DLOG(WARNING) << kAlternateProtocolHeader
<< " header has too many tokens: "
<< alternate_protocol_str;
return;
}
int port;
if (!base::StringToInt(port_protocol_vector[0], &port) ||
port <= 0 || port >= 1 << 16) {
DLOG(WARNING) << kAlternateProtocolHeader
<< " header has unrecognizable port: "
<< port_protocol_vector[0];
return;
}
AlternateProtocol protocol = ALTERNATE_PROTOCOL_BROKEN;
for (int i = 0; i < NUM_ALTERNATE_PROTOCOLS; ++i) {
if (enabled_protocols_[i] &&
port_protocol_vector[1] == kAlternateProtocolStrings[i]) {
protocol = static_cast<AlternateProtocol>(i);
}
}
if (protocol == ALTERNATE_PROTOCOL_BROKEN) {
// This warning is not important for LbShell so we are going to disable it
// since it produces considerable console spam
#if !defined(__LB_SHELL__) && !defined(COBALT)
// Currently, we only recognize the npn-spdy protocol.
DLOG(WARNING) << kAlternateProtocolHeader
<< " header has unrecognized protocol: "
<< port_protocol_vector[1];
#endif
return;
}
HostPortPair host_port(http_host_port_pair);
const HostMappingRules* mapping_rules = GetHostMappingRules();
if (mapping_rules)
mapping_rules->RewriteHost(&host_port);
if (http_server_properties->HasAlternateProtocol(host_port)) {
const PortAlternateProtocolPair existing_alternate =
http_server_properties->GetAlternateProtocol(host_port);
// If we think the alternate protocol is broken, don't change it.
if (existing_alternate.protocol == ALTERNATE_PROTOCOL_BROKEN)
return;
}
http_server_properties->SetAlternateProtocol(host_port, port, protocol);
}
GURL HttpStreamFactory::ApplyHostMappingRules(const GURL& url,
HostPortPair* endpoint) {
const HostMappingRules* mapping_rules = GetHostMappingRules();
if (mapping_rules && mapping_rules->RewriteHost(endpoint)) {
url_canon::Replacements<char> replacements;
const std::string port_str = base::IntToString(endpoint->port());
replacements.SetPort(port_str.c_str(),
url_parse::Component(0, port_str.size()));
replacements.SetHost(endpoint->host().c_str(),
url_parse::Component(0, endpoint->host().size()));
return url.ReplaceComponents(replacements);
}
return url;
}
// static
void HttpStreamFactory::add_forced_spdy_exclusion(const std::string& value) {
HostPortPair pair = HostPortPair::FromURL(GURL(value));
if (!forced_spdy_exclusions_)
forced_spdy_exclusions_ = new std::list<HostPortPair>();
forced_spdy_exclusions_->push_back(pair);
}
// static
bool HttpStreamFactory::HasSpdyExclusion(const HostPortPair& endpoint) {
std::list<HostPortPair>* exclusions = forced_spdy_exclusions_;
if (!exclusions)
return false;
std::list<HostPortPair>::const_iterator it;
for (it = exclusions->begin(); it != exclusions->end(); ++it)
if (it->Equals(endpoint))
return true;
return false;
}
// static
void HttpStreamFactory::EnableNpnSpdy() {
set_use_alternate_protocols(true);
std::vector<std::string> next_protos;
next_protos.push_back("http/1.1");
next_protos.push_back("spdy/2");
SetNextProtos(next_protos);
}
// static
void HttpStreamFactory::EnableNpnHttpOnly() {
// Avoid alternate protocol in this case. Otherwise, browser will try SSL
// and then fallback to http. This introduces extra load.
set_use_alternate_protocols(false);
std::vector<std::string> next_protos;
next_protos.push_back("http/1.1");
next_protos.push_back("http1.1");
SetNextProtos(next_protos);
}
// static
void HttpStreamFactory::EnableNpnSpdy3() {
set_use_alternate_protocols(true);
std::vector<std::string> next_protos;
next_protos.push_back("http/1.1");
next_protos.push_back("spdy/2");
next_protos.push_back("spdy/3");
SetNextProtos(next_protos);
}
// static
void HttpStreamFactory::SetNextProtos(const std::vector<std::string>& value) {
if (!next_protos_)
next_protos_ = new std::vector<std::string>;
*next_protos_ = value;
for (uint32 i = 0; i < NUM_ALTERNATE_PROTOCOLS; ++i)
enabled_protocols_[i] = false;
// TODO(rtenneti): bug 116575 - consider using same strings/enums for SPDY
// versions in next_protos and kAlternateProtocolStrings.
for (uint32 i = 0; i < value.size(); ++i) {
if (value[i] == "spdy/1") {
enabled_protocols_[NPN_SPDY_1] = true;
} else if (value[i] == "spdy/2") {
enabled_protocols_[NPN_SPDY_2] = true;
} else if (value[i] == "spdy/3") {
enabled_protocols_[NPN_SPDY_3] = true;
}
}
enabled_protocols_[NPN_SPDY_1] = false;
}
HttpStreamFactory::HttpStreamFactory() {}
} // namespace net