Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

rds: expose path match criterion via route entry #2531

Merged
merged 4 commits into from
Feb 6, 2018
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
33 changes: 33 additions & 0 deletions include/envoy/router/router.h
Original file line number Diff line number Diff line change
Expand Up @@ -306,6 +306,34 @@ class MetadataMatchCriteria {
metadataMatchCriteria() const PURE;
};

/**
* Type of path matching that a route entry uses.
*/
enum class PathMatchType {
None,
Prefix,
Exact,
Regex,
};

/**
* Criterion that a route entry uses for matching a particular path.
*/
class PathMatchCriterion {
public:
virtual ~PathMatchCriterion() {}

/**
* @return PathMatchType type of path match.
*/
virtual PathMatchType matchType() const PURE;

/**
* @return const std::string& the string with which to compare paths.
*/
virtual const std::string& matcher() const PURE;
};

/**
* An individual resolved route entry.
*/
Expand Down Expand Up @@ -415,6 +443,11 @@ class RouteEntry : public ResponseEntry {
* this route.
*/
virtual const envoy::api::v2::core::Metadata& metadata() const PURE;

/**
* @return const PathMatchCriterion& the match criterion for this route.
*/
virtual const PathMatchCriterion& pathMatchCriterion() const PURE;
};

/**
Expand Down
2 changes: 2 additions & 0 deletions source/common/http/async_client_impl.cc
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,8 @@ const AsyncStreamImpl::NullVirtualHost AsyncStreamImpl::RouteEntryImpl::virtual_
const AsyncStreamImpl::NullRateLimitPolicy AsyncStreamImpl::NullVirtualHost::rate_limit_policy_;
const std::multimap<std::string, std::string> AsyncStreamImpl::RouteEntryImpl::opaque_config_;
const envoy::api::v2::core::Metadata AsyncStreamImpl::RouteEntryImpl::metadata_;
const AsyncStreamImpl::NullPathMatchCriterion
AsyncStreamImpl::RouteEntryImpl::path_match_criterion_;

AsyncClientImpl::AsyncClientImpl(const Upstream::ClusterInfo& cluster, Stats::Store& stats_store,
Event::Dispatcher& dispatcher,
Expand Down
9 changes: 9 additions & 0 deletions source/common/http/async_client_impl.h
Original file line number Diff line number Diff line change
Expand Up @@ -135,6 +135,11 @@ class AsyncStreamImpl : public AsyncClient::Stream,
static const NullRateLimitPolicy rate_limit_policy_;
};

struct NullPathMatchCriterion : public Router::PathMatchCriterion {
Router::PathMatchType matchType() const override { return Router::PathMatchType::None; }
const std::string& matcher() const override { return EMPTY_STRING; }
};

struct RouteEntryImpl : public Router::RouteEntry {
RouteEntryImpl(const std::string& cluster_name,
const Optional<std::chrono::milliseconds>& timeout)
Expand Down Expand Up @@ -175,13 +180,17 @@ class AsyncStreamImpl : public AsyncClient::Stream,
bool useWebSocket() const override { return false; }
bool includeVirtualHostRateLimits() const override { return true; }
const envoy::api::v2::core::Metadata& metadata() const override { return metadata_; }
const Router::PathMatchCriterion& pathMatchCriterion() const override {
return path_match_criterion_;
}

static const NullRateLimitPolicy rate_limit_policy_;
static const NullRetryPolicy retry_policy_;
static const NullShadowPolicy shadow_policy_;
static const NullVirtualHost virtual_host_;
static const std::multimap<std::string, std::string> opaque_config_;
static const envoy::api::v2::core::Metadata metadata_;
static const NullPathMatchCriterion path_match_criterion_;

const std::string& cluster_name_;
Optional<std::chrono::milliseconds> timeout_;
Expand Down
3 changes: 2 additions & 1 deletion source/common/router/config_impl.cc
Original file line number Diff line number Diff line change
Expand Up @@ -592,7 +592,8 @@ RegexRouteEntryImpl::RegexRouteEntryImpl(const VirtualHostImpl& vhost,
const envoy::api::v2::route::Route& route,
Runtime::Loader& loader)
: RouteEntryImplBase(vhost, route, loader),
regex_(RegexUtil::parseRegex(route.match().regex().c_str())) {}
regex_(RegexUtil::parseRegex(route.match().regex().c_str())),
regex_str_(route.match().regex()) {}

void RegexRouteEntryImpl::finalizeRequestHeaders(
Http::HeaderMap& headers, const RequestInfo::RequestInfo& request_info) const {
Expand Down
18 changes: 18 additions & 0 deletions source/common/router/config_impl.h
Original file line number Diff line number Diff line change
Expand Up @@ -290,6 +290,7 @@ class RouteEntryImplBase : public RouteEntry,
public Matchable,
public DirectResponseEntry,
public Route,
public PathMatchCriterion,
public std::enable_shared_from_this<RouteEntryImplBase> {
public:
/**
Expand Down Expand Up @@ -334,6 +335,7 @@ class RouteEntryImplBase : public RouteEntry,
}
bool includeVirtualHostRateLimits() const override { return include_vh_rate_limits_; }
const envoy::api::v2::core::Metadata& metadata() const override { return metadata_; }
const PathMatchCriterion& pathMatchCriterion() const override { return *this; }
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm wondering if I've forgotten how to read C++ :) Can you explain why this is needed (or even how it passes type checking?).

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, so the inheritance structure is a little convoluted here. RouteEntryImplBase inherits from RouteEntry, Matchable, and now PathMatchCriterion (among others). It has three derived classes (PrefixRouteEntryImpl, PathRouteEntryImpl, and RegexRouteEntryImpl) that implement the Matchable and PathMatchCriterion abstract methods since they implement all of the match-type specific logic. So, for the pathMatchCriterion() method (part of the RouteEntry interface), it just returns itself as the PathMatchCriterion since it implements that interface. Does that help clarify things?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does it ever need to be directly instantiated (i.e. be concrete)? Or do we only use derived classes? If the latter, it seems we don't need to define it.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The intent of the class was to group the idea of a route match type and a matcher since one is needed to interpret the other (like a virtualized POD struct) - as opposed to polluting the RouteEntry interface (which is already starting to feel a little big) with a routeMatchType() method and a routeMatchString() method or something. I'm happy to promote them to the RouteEntry interface if that seems like a net gain, though.


// Router::DirectResponseEntry
std::string newPath(const Http::HeaderMap& headers) const override;
Expand Down Expand Up @@ -408,6 +410,9 @@ class RouteEntryImplBase : public RouteEntry,
return parent_->includeVirtualHostRateLimits();
}
const envoy::api::v2::core::Metadata& metadata() const override { return parent_->metadata(); }
const PathMatchCriterion& pathMatchCriterion() const override {
return parent_->pathMatchCriterion();
}

// Router::Route
const DirectResponseEntry* directResponseEntry() const override { return nullptr; }
Expand Down Expand Up @@ -512,6 +517,10 @@ class PrefixRouteEntryImpl : public RouteEntryImplBase {
void finalizeRequestHeaders(Http::HeaderMap& headers,
const RequestInfo::RequestInfo& request_info) const override;

// Router::PathMatchCriterion
const std::string& matcher() const override { return prefix_; }
PathMatchType matchType() const override { return PathMatchType::Prefix; }

// Router::Matchable
RouteConstSharedPtr matches(const Http::HeaderMap& headers, uint64_t random_value) const override;

Expand All @@ -531,6 +540,10 @@ class PathRouteEntryImpl : public RouteEntryImplBase {
void finalizeRequestHeaders(Http::HeaderMap& headers,
const RequestInfo::RequestInfo& request_info) const override;

// Router::PathMatchCriterion
const std::string& matcher() const override { return path_; }
PathMatchType matchType() const override { return PathMatchType::Exact; }

// Router::Matchable
RouteConstSharedPtr matches(const Http::HeaderMap& headers, uint64_t random_value) const override;

Expand All @@ -550,11 +563,16 @@ class RegexRouteEntryImpl : public RouteEntryImplBase {
void finalizeRequestHeaders(Http::HeaderMap& headers,
const RequestInfo::RequestInfo& request_info) const override;

// Router::PathMatchCriterion
const std::string& matcher() const override { return regex_str_; }
PathMatchType matchType() const override { return PathMatchType::Regex; }

// Router::Matchable
RouteConstSharedPtr matches(const Http::HeaderMap& headers, uint64_t random_value) const override;

private:
const std::regex regex_;
const std::string regex_str_;
};

/**
Expand Down
18 changes: 18 additions & 0 deletions test/common/http/async_client_impl_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -863,6 +863,24 @@ TEST_F(AsyncClientImplTest, WatermarkCallbacks) {
EXPECT_CALL(stream_callbacks_, onReset());
}

TEST_F(AsyncClientImplTest, NullPathMatchCriterion) {
TestHeaderMapImpl headers;
HttpTestUtility::addDefaultHeaders(headers);
AsyncClient::Stream* stream =
client_.start(stream_callbacks_, Optional<std::chrono::milliseconds>(), false);
stream->sendHeaders(headers, false);
Http::StreamDecoderFilterCallbacks* filter_callbacks =
static_cast<Http::AsyncStreamImpl*>(stream);
auto route = filter_callbacks->route();
ASSERT_NE(nullptr, route);
auto route_entry = route->routeEntry();
ASSERT_NE(nullptr, route_entry);
auto& path_match_criterion = route_entry->pathMatchCriterion();
EXPECT_EQ("", path_match_criterion.matcher());
EXPECT_EQ(Router::PathMatchType::None, path_match_criterion.matchType());
EXPECT_CALL(stream_callbacks_, onReset());
}

} // namespace
} // namespace Http
} // namespace Envoy
25 changes: 22 additions & 3 deletions test/common/router/config_impl_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -3773,13 +3773,27 @@ name: foo
EnvoyException, "response body size is 4097 bytes; maximum is 4096");
}

TEST(RouteConfigurationV2, Metadata) {
void checkPathMatchCriterion(const Route* route, const std::string& expected_matcher,
PathMatchType expected_type) {
ASSERT_NE(nullptr, route);
auto route_entry = route->routeEntry();
ASSERT_NE(nullptr, route_entry);
auto& match_criterion = route_entry->pathMatchCriterion();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nit: const etc.

EXPECT_EQ(expected_matcher, match_criterion.matcher());
EXPECT_EQ(expected_type, match_criterion.matchType());
}

TEST(RouteConfigurationV2, RouteConfigGetters) {
std::string yaml = R"EOF(
name: foo
virtual_hosts:
- name: bar
domains: ["*"]
routes:
- match: { regex: "/rege[xy]" }
route: { cluster: ww2 }
- match: { path: "/exact-path" }
route: { cluster: ww2 }
- match: { prefix: "/"}
route: { cluster: www2 }
metadata: { filter_metadata: { com.bar.foo: { baz: test_value } } }
Expand All @@ -3789,9 +3803,14 @@ name: foo
NiceMock<Upstream::MockClusterManager> cm;
ConfigImpl config(parseRouteConfigurationFromV2Yaml(yaml), runtime, cm, true);

auto* route_entry = config.route(genHeaders("www.foo.com", "/", "GET"), 0)->routeEntry();
checkPathMatchCriterion(config.route(genHeaders("www.foo.com", "/regex", "GET"), 0).get(),
"/rege[xy]", PathMatchType::Regex);
checkPathMatchCriterion(config.route(genHeaders("www.foo.com", "/exact-path", "GET"), 0).get(),
"/exact-path", PathMatchType::Exact);
auto route = config.route(genHeaders("www.foo.com", "/", "GET"), 0);
checkPathMatchCriterion(route.get(), "/", PathMatchType::Prefix);

const auto& metadata = route_entry->metadata();
const auto& metadata = route->routeEntry()->metadata();

EXPECT_EQ("test_value",
Envoy::Config::Metadata::metadataValue(metadata, "com.bar.foo", "baz").string_value());
Expand Down
1 change: 1 addition & 0 deletions test/mocks/router/mocks.h
Original file line number Diff line number Diff line change
Expand Up @@ -216,6 +216,7 @@ class MockRouteEntry : public RouteEntry {
MOCK_CONST_METHOD0(includeVirtualHostRateLimits, bool());
MOCK_CONST_METHOD0(corsPolicy, const CorsPolicy*());
MOCK_CONST_METHOD0(metadata, const envoy::api::v2::core::Metadata&());
MOCK_CONST_METHOD0(pathMatchCriterion, const PathMatchCriterion&());

std::string cluster_name_{"fake_cluster"};
std::multimap<std::string, std::string> opaque_config_;
Expand Down