-
Notifications
You must be signed in to change notification settings - Fork 89
Adding config factory #522
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
Changes from all commits
99f45d1
075a4cf
ecbf407
8fb11b6
032dbba
953e953
6185edb
cded6ad
fbfe523
0d70179
cdb7efe
f15bb47
59359a2
f63df91
406bd6b
f0b697b
8425e48
fb4f372
c6eae6c
96ae38b
b17240f
5c5a146
29591ff
7d928a9
ff82d95
98381a4
83ac556
38c5084
fa4e285
8551c76
437de5e
6cef87e
c84ba4d
77d33f9
73aa686
ef77d65
b9182a0
e38dd80
21734e4
7cda9d2
f827aca
d194736
7820cf9
bee7805
3db68ce
c0ec4dd
a1dd3f2
0b25261
c3fa23c
547977e
c300203
8f4d4d9
5ae2d51
fe74b6d
6d87aea
ca7a0c2
967e3f8
14b7d79
c4985bb
922639a
81dfece
bf08661
34900c5
768fa81
e93bb58
f47cbd4
7c5c6b2
1b02cec
4d75602
c74f535
9949fb8
fcf10d3
fd9744d
68c1037
883ee30
8c34b25
6ce7e74
92927e5
a62f2d5
16f89c6
f4257e3
d075588
eb10a3b
cdbd6bf
17acc5f
55a0e62
fa23aad
b8e2447
248f439
91703b5
bbc0c2a
517c83d
c1d0428
bc86c09
33e3564
338a874
402ada2
017481f
3ed43fc
247b71b
171fb2e
5df83ed
0af3ffd
ff90e59
e57c05f
55244b0
c710d37
fbb2fdc
ae0f4a8
e092de0
19b7aa0
dae3098
a4b6fb1
55f685e
59c5b94
2ea0c5e
d40a073
7146079
7ab1132
72c23d0
2b1170c
9dc3c4b
b555a71
3924eee
75eca07
9c93ff9
e2af886
edae9be
949d731
59d490f
078329a
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,36 @@ | ||
| // Config protos for the Request Source Plugin Config Factories. | ||
| syntax = "proto3"; | ||
|
|
||
| package nighthawk.request_source; | ||
|
|
||
| import "google/protobuf/wrappers.proto"; | ||
| import "validate/validate.proto"; | ||
| import "api/client/options.proto"; | ||
|
|
||
| // Configuration for FileBasedPluginRequestSource (plugin name: | ||
| // "nighthawk.file-based-request-source-plugin") | ||
| // The factory will load the RequestOptionsList from the file, and then passes it to the | ||
| // requestSource it generates. The resulting request source will loop over the RequestOptionsList it | ||
| // is passed. | ||
| message FileBasedPluginConfig { | ||
| // The file_path is the path to a file that contains a RequestOptionList in json or yaml format. | ||
| string file_path = 1; | ||
| // The pluginfactory makes requestSources that will generate requests from the RequestOptionList | ||
| // up to num_requests number of times. If num_requests exceeds the number of RequestOptions in the | ||
| // RequestOptionList located in the file at file_path, it will loop. num_requests = 0 means no | ||
| // limit on the number of requests to be produced. | ||
| google.protobuf.UInt32Value num_requests = 2 [(validate.rules).uint32 = {gte: 0, lte: 1000000}]; | ||
| // The pluginfactory will load the file located in file_path as long as it is below max_file_size, | ||
| // if it's too large it will throw an error. | ||
| google.protobuf.UInt32Value max_file_size = 3 [(validate.rules).uint32 = {lte: 1000000}]; | ||
| } | ||
|
|
||
| // Configuration for StubPluginRequestSource (plugin name: "nighthawk.stub-request-source-plugin") | ||
| // The plugin does nothing. This is for testing and comparison of the Request Source Plugin Factory | ||
| // mechanism using a minimal version of plugin that does not require a more complicated proto or | ||
| // file reading. | ||
| message StubPluginConfig { | ||
| // test input value which is the only output value in the headers produced from the | ||
| // requestGenerator for the StubRequestSource. | ||
| google.protobuf.DoubleValue test_value = 1; | ||
| } |
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,24 @@ | ||
| load( | ||
| "@envoy//bazel:envoy_build_system.bzl", | ||
| "envoy_basic_cc_library", | ||
| "envoy_package", | ||
| ) | ||
|
|
||
| licenses(["notice"]) # Apache 2 | ||
|
|
||
| envoy_package() | ||
|
|
||
| envoy_basic_cc_library( | ||
| name = "request_source_plugin_config_factory_lib", | ||
| hdrs = [ | ||
| "request_source_plugin_config_factory.h", | ||
| ], | ||
| include_prefix = "nighthawk/request_source", | ||
| deps = [ | ||
| "//api/request_source:request_source_plugin_cc_proto", | ||
| "//include/nighthawk/common:request_source_lib", | ||
| "@envoy//include/envoy/common:base_includes", | ||
| "@envoy//include/envoy/config:typed_config_interface", | ||
| "@envoy//source/common/api:api_lib_with_external_headers", | ||
| ], | ||
| ) |
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,39 @@ | ||
| #pragma once | ||
|
|
||
| #include "envoy/api/api.h" | ||
| #include "envoy/common/pure.h" | ||
| #include "envoy/config/typed_config.h" | ||
|
|
||
| #include "nighthawk/common/request_source.h" | ||
|
|
||
| namespace Nighthawk { | ||
|
|
||
| // A factory that must be implemented for each RequestSourcePlugin. It instantiates the specific | ||
| // RequestSourcePlugin class after unpacking the plugin-specific config proto. | ||
| class RequestSourcePluginConfigFactory : public Envoy::Config::TypedFactory { | ||
| public: | ||
| ~RequestSourcePluginConfigFactory() override = default; | ||
| // All request source plugins will be in this category. | ||
| std::string category() const override { return "nighthawk.request_source_plugin"; } | ||
|
|
||
| // Instantiates the specific RequestSourcePlugin class. Casts |message| to Any, unpacks it to the | ||
| // plugin-specific proto, and passes the strongly typed proto to the plugin constructor. | ||
| // | ||
| // @param typed_config Any typed_config proto taken from the TypedExtensionConfig. This should be | ||
| // a type listed in request_source_plugin_config.proto | ||
| // | ||
| // @param api Api parameter that contains timesystem, filesystem, and threadfactory. | ||
| // | ||
| // @param header RequestHeaderMapPtr parameter that acts as a template header for the | ||
| // requestSource to modify when generating requests. | ||
| // | ||
| // @return RequestSourcePtr Pointer to the new instance of RequestSource. | ||
| // | ||
| // @throw Envoy::EnvoyException If the Any proto cannot be unpacked as the type expected by the | ||
| // plugin. | ||
| virtual RequestSourcePtr createRequestSourcePlugin(const Envoy::Protobuf::Message& typed_config, | ||
| Envoy::Api::Api& api, | ||
| Envoy::Http::RequestHeaderMapPtr header) PURE; | ||
| }; | ||
|
|
||
| } // namespace Nighthawk |
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,33 @@ | ||
| load( | ||
| "@envoy//bazel:envoy_build_system.bzl", | ||
| "envoy_cc_library", | ||
| "envoy_package", | ||
| ) | ||
|
|
||
| licenses(["notice"]) # Apache 2 | ||
|
|
||
| envoy_package() | ||
|
|
||
| envoy_cc_library( | ||
| name = "request_options_list_plugin_impl", | ||
| srcs = [ | ||
| "request_options_list_plugin_impl.cc", | ||
|
Collaborator
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The choice of file name here doesn't directly relate to what it contains. Looks like it contains the File Based Request source. Can we call the file accordingly, e.g. "file_based_plugin.cc" or "file_based_request_source.cc". We should also adjust the name of the BUILD target to match. EDIT: I noticed that the file contains multiple request source implementations. If we end up keeping the RequestOptionsListRequestSource, let's have it in its own file and library also.
Contributor
Author
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Understood. I wanted this to be called the RequestOptionsListPluginImpl because I figured both the FileBasedPluginConfigFactory, and the RequestOptionListPluginConfigFactory would make the same RequestOptionListPlugin, just loading from a different location. |
||
| ], | ||
| hdrs = [ | ||
| "request_options_list_plugin_impl.h", | ||
| ], | ||
| repository = "@envoy", | ||
| visibility = ["//visibility:public"], | ||
| deps = [ | ||
| "//include/nighthawk/request_source:request_source_plugin_config_factory_lib", | ||
| "//source/common:nighthawk_common_lib", | ||
| "//source/common:request_impl_lib", | ||
| "//source/common:request_source_impl_lib", | ||
| "@envoy//source/common/common:thread_lib_with_external_headers", | ||
| "@envoy//source/common/protobuf:message_validator_lib_with_external_headers", | ||
| "@envoy//source/common/protobuf:protobuf_with_external_headers", | ||
| "@envoy//source/common/protobuf:utility_lib_with_external_headers", | ||
| "@envoy//source/exe:platform_header_lib_with_external_headers", | ||
| "@envoy//source/exe:platform_impl_lib", | ||
| ], | ||
| ) | ||
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,89 @@ | ||
| #include "request_source/request_options_list_plugin_impl.h" | ||
|
|
||
| #include "external/envoy/source/common/protobuf/message_validator_impl.h" | ||
| #include "external/envoy/source/common/protobuf/utility.h" | ||
| #include "external/envoy/source/exe/platform_impl.h" | ||
|
|
||
| #include "api/client/options.pb.h" | ||
|
|
||
| #include "common/request_impl.h" | ||
| #include "common/request_source_impl.h" | ||
|
|
||
| namespace Nighthawk { | ||
| std::string OptionsListFromFileRequestSourceFactory::name() const { | ||
| return "nighthawk.file-based-request-source-plugin"; | ||
| } | ||
|
|
||
| Envoy::ProtobufTypes::MessagePtr OptionsListFromFileRequestSourceFactory::createEmptyConfigProto() { | ||
| return std::make_unique<nighthawk::request_source::FileBasedPluginConfig>(); | ||
| } | ||
|
|
||
| RequestSourcePtr OptionsListFromFileRequestSourceFactory::createRequestSourcePlugin( | ||
| const Envoy::Protobuf::Message& message, Envoy::Api::Api& api, | ||
| Envoy::Http::RequestHeaderMapPtr header) { | ||
| const auto& any = dynamic_cast<const Envoy::ProtobufWkt::Any&>(message); | ||
| nighthawk::request_source::FileBasedPluginConfig config; | ||
| Envoy::MessageUtil util; | ||
|
|
||
| util.unpackTo(any, config); | ||
| if (api.fileSystem().fileSize(config.file_path()) > config.max_file_size().value()) { | ||
| throw NighthawkException("file size must be less than max_file_size"); | ||
| } | ||
|
|
||
| // Locking to avoid issues with multiple threads reading the same file. | ||
| { | ||
| Envoy::Thread::LockGuard lock_guard(file_lock_); | ||
| // Reading the file only the first time. | ||
| if (options_list_.options_size() == 0) { | ||
| util.loadFromFile(config.file_path(), options_list_, | ||
| Envoy::ProtobufMessage::getStrictValidationVisitor(), api, true); | ||
| } | ||
| } | ||
| return std::make_unique<RequestOptionsListRequestSource>(config.num_requests().value(), | ||
| std::move(header), options_list_); | ||
| } | ||
|
|
||
| REGISTER_FACTORY(OptionsListFromFileRequestSourceFactory, RequestSourcePluginConfigFactory); | ||
|
|
||
| RequestOptionsListRequestSource::RequestOptionsListRequestSource( | ||
| const uint32_t total_requests, Envoy::Http::RequestHeaderMapPtr header, | ||
| const nighthawk::client::RequestOptionsList& options_list) | ||
| : header_(std::move(header)), options_list_(options_list), total_requests_(total_requests) {} | ||
|
|
||
| RequestGenerator RequestOptionsListRequestSource::get() { | ||
| request_count_.push_back(0); | ||
| uint32_t& lambda_counter = request_count_.back(); | ||
| RequestGenerator request_generator = [this, lambda_counter]() mutable -> RequestPtr { | ||
|
Collaborator
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can you help me understand the need for the "mutable" keyword here?
Contributor
Author
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I need this keyword because if I do not have mutable, the compiler will complain about the incrementing of the lambda_counter later in this lambda. |
||
| // if request_max is 0, then we never stop generating requests. | ||
| if (lambda_counter >= total_requests_ && total_requests_ != 0) { | ||
| return nullptr; | ||
| } | ||
|
|
||
| // Increment the counter and get the request_option from the list for the current iteration. | ||
| const uint32_t index = lambda_counter % options_list_.options_size(); | ||
| nighthawk::client::RequestOptions request_option = options_list_.options().at(index); | ||
| ++lambda_counter; | ||
|
|
||
| // Initialize the header with the values from the default header. | ||
| Envoy::Http::RequestHeaderMapPtr header = Envoy::Http::RequestHeaderMapImpl::create(); | ||
| Envoy::Http::HeaderMapImpl::copyFrom(*header, *header_); | ||
|
Collaborator
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Since we are making a copy of the original header map anyway, why do we need to pass it in as a pointer to the plugin? Can't the plugin take the header as a const reference to make it clear on the API layer that the header isn't owned or modified? Or is there a reason that forces us to give ownership?
Contributor
Author
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think that it is more consistent with the way that the other plugins do it, and it is possible that the plugin would want to modify the header. Generally it also seems like something that the RequestSource should own. In the factories_impl.cc where this RequestSource will be created, I believe the header passed in does not have a lifetime greater than that of the RequestSource, and that might matter also, though I'm not sure if that is correct. |
||
|
|
||
| // Override the default values with the values from the request_option | ||
| header->setMethod(envoy::config::core::v3::RequestMethod_Name(request_option.request_method())); | ||
| const uint32_t content_length = request_option.request_body_size().value(); | ||
| if (content_length > 0) { | ||
| header->setContentLength(content_length); | ||
| } | ||
| for (const envoy::config::core::v3::HeaderValueOption& option_header : | ||
| request_option.request_headers()) { | ||
| auto lower_case_key = Envoy::Http::LowerCaseString(std::string(option_header.header().key())); | ||
| header->setCopy(lower_case_key, std::string(option_header.header().value())); | ||
| } | ||
| return std::make_unique<RequestImpl>(std::move(header)); | ||
| }; | ||
| return request_generator; | ||
| } | ||
|
|
||
| void RequestOptionsListRequestSource::initOnThread() {} | ||
|
|
||
| } // namespace Nighthawk | ||
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,84 @@ | ||
| // Implementations of RequestSourceConfigFactories that make a RequestOptionsListRequestSource. | ||
| #pragma once | ||
|
|
||
| #include "envoy/registry/registry.h" | ||
|
|
||
| #include "nighthawk/request_source/request_source_plugin_config_factory.h" | ||
|
|
||
| #include "external/envoy/source/common/common/lock_guard.h" | ||
| #include "external/envoy/source/common/common/thread.h" | ||
|
|
||
| #include "api/client/options.pb.h" | ||
| #include "api/request_source/request_source_plugin.pb.h" | ||
|
|
||
| #include "common/uri_impl.h" | ||
|
|
||
| namespace Nighthawk { | ||
|
|
||
| // Sample Request Source for small RequestOptionsLists. Loads a copy of the RequestOptionsList in | ||
| // memory and replays them. | ||
| // @param total_requests The number of requests the requestGenerator produced by get() will | ||
| // generate. 0 means it is unlimited. | ||
| // @param header the default header that will be overridden by values taken from the options_list, | ||
| // any values not overridden will be used. | ||
| // @param options_list This is const because it is intended to be shared by multiple threads. The | ||
| // RequestGenerator produced by get() will use options from the options_list to overwrite values in | ||
| // the default header, and create new requests. if total_requests is greater than the length of | ||
| // options_list, it will loop. This is not thread safe. | ||
| class RequestOptionsListRequestSource : public RequestSource { | ||
| public: | ||
| RequestOptionsListRequestSource(const uint32_t total_requests, | ||
| Envoy::Http::RequestHeaderMapPtr header, | ||
| const nighthawk::client::RequestOptionsList& options_list); | ||
|
|
||
| // This get function is not thread safe, because multiple threads calling get simultaneously will | ||
| // result in a collision as it attempts to update its request_count_. | ||
| RequestGenerator get() override; | ||
|
|
||
| // default implementation | ||
| void initOnThread() override; | ||
|
|
||
| private: | ||
| Envoy::Http::RequestHeaderMapPtr header_; | ||
| const nighthawk::client::RequestOptionsList& options_list_; | ||
| std::vector<uint32_t> request_count_; | ||
| const uint32_t total_requests_; | ||
| }; | ||
|
|
||
| // Factory that creates a RequestOptionsListRequestSource from a FileBasedPluginConfig proto. | ||
| // Registered as an Envoy plugin. | ||
| // Implementation of RequestSourceConfigFactory which produces a RequestSource that keeps an | ||
| // RequestOptionsList in memory, and loads it with the RequestOptions taken from a file. All plugins | ||
| // configuration are specified in the request_source_plugin.proto. This class is not thread-safe, | ||
| // because it loads its RequestOptionlist in memory from a file when first called. | ||
| // Usage: assume you are passed an appropriate Any type object called config, an Api object called | ||
| // api, and a default header called header. auto& config_factory = | ||
| // Envoy::Config::Utility::getAndCheckFactoryByName<RequestSourcePluginConfigFactory>( | ||
| // "nighthawk.file-based-request-source-plugin"); | ||
| // RequestSourcePtr plugin = | ||
| // config_factory.createRequestSourcePlugin(config, std::move(api), std::move(header)); | ||
| class OptionsListFromFileRequestSourceFactory : public virtual RequestSourcePluginConfigFactory { | ||
| public: | ||
| std::string name() const override; | ||
|
|
||
| Envoy::ProtobufTypes::MessagePtr createEmptyConfigProto() override; | ||
|
|
||
| // This implementation is not thread safe. Only the first call to createRequestSourcePlugin will | ||
| // load the file from memory and subsequent calls just make a copy of the options_list that was | ||
| // already loaded. The OptionsListFromFileRequestSourceFactory will not work with multiple | ||
| // different files for this reason. | ||
| // This method will also error if the file can not be loaded correctly, e.g. the file is too large | ||
| // or could not be found. | ||
| RequestSourcePtr createRequestSourcePlugin(const Envoy::Protobuf::Message& message, | ||
| Envoy::Api::Api& api, | ||
| Envoy::Http::RequestHeaderMapPtr header) override; | ||
|
|
||
| private: | ||
| Envoy::Thread::MutexBasicLockable file_lock_; | ||
| nighthawk::client::RequestOptionsList options_list_; | ||
| }; | ||
|
|
||
| // This factory will be activated through RequestSourceFactory in factories.h | ||
| DECLARE_FACTORY(OptionsListFromFileRequestSourceFactory); | ||
|
|
||
| } // namespace Nighthawk |
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,48 @@ | ||
| load( | ||
| "@envoy//bazel:envoy_build_system.bzl", | ||
| "envoy_cc_test", | ||
| "envoy_cc_test_library", | ||
| "envoy_package", | ||
| ) | ||
|
|
||
| licenses(["notice"]) # Apache 2 | ||
|
|
||
| envoy_package() | ||
|
|
||
| envoy_cc_test_library( | ||
| name = "stub_plugin_impl", | ||
| srcs = [ | ||
| "stub_plugin_impl.cc", | ||
| ], | ||
| hdrs = [ | ||
| "stub_plugin_impl.h", | ||
| ], | ||
| repository = "@envoy", | ||
| deps = [ | ||
| "//include/nighthawk/request_source:request_source_plugin_config_factory_lib", | ||
| "//source/common:nighthawk_common_lib", | ||
| "//source/common:request_impl_lib", | ||
| "//source/common:request_source_impl_lib", | ||
| "@envoy//source/common/protobuf:message_validator_lib_with_external_headers", | ||
| "@envoy//source/common/protobuf:protobuf_with_external_headers", | ||
| "@envoy//source/common/protobuf:utility_lib_with_external_headers", | ||
| "@envoy//source/exe:platform_header_lib_with_external_headers", | ||
| "@envoy//source/exe:platform_impl_lib", | ||
| ], | ||
| ) | ||
|
|
||
| envoy_cc_test( | ||
| name = "request_source_plugin_test", | ||
| srcs = ["request_source_plugin_test.cc"], | ||
| data = [ | ||
| "test_data/test-config.yaml", | ||
| ], | ||
| repository = "@envoy", | ||
| deps = [ | ||
| "//source/request_source:request_options_list_plugin_impl", | ||
| "//test/request_source:stub_plugin_impl", | ||
| "//test/test_common:environment_lib", | ||
| "@envoy//source/common/config:utility_lib_with_external_headers", | ||
| "@envoy//test/mocks/api:api_mocks", | ||
| ], | ||
| ) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Plz add comment for the proto and its field.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can do.