summaryrefslogtreecommitdiffstats
path: root/src/seastar/tests/unit/chunk_parsers_test.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/seastar/tests/unit/chunk_parsers_test.cc')
-rw-r--r--src/seastar/tests/unit/chunk_parsers_test.cc120
1 files changed, 120 insertions, 0 deletions
diff --git a/src/seastar/tests/unit/chunk_parsers_test.cc b/src/seastar/tests/unit/chunk_parsers_test.cc
new file mode 100644
index 000000000..e15b01f04
--- /dev/null
+++ b/src/seastar/tests/unit/chunk_parsers_test.cc
@@ -0,0 +1,120 @@
+/*
+ * This file is open source software, licensed to you under the terms
+ * of the Apache License, Version 2.0 (the "License"). See the NOTICE file
+ * distributed with this work for additional information regarding copyright
+ * ownership. You may not use this file except in compliance with the License.
+ *
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+/*
+ * Copyright (C) 2020 ScyllaDB.
+ */
+
+#include <seastar/core/ragel.hh>
+#include <seastar/core/sstring.hh>
+#include <seastar/core/thread.hh>
+#include <seastar/core/temporary_buffer.hh>
+#include <seastar/http/chunk_parsers.hh>
+#include <seastar/http/internal/content_source.hh>
+#include <seastar/testing/test_case.hh>
+#include <tuple>
+#include <utility>
+#include <vector>
+
+using namespace seastar;
+
+SEASTAR_TEST_CASE(test_size_and_extensions_parsing) {
+ struct test_set {
+ sstring msg;
+ bool parsable;
+ sstring size = "";
+ std::vector<std::pair<sstring, sstring>> extensions;
+
+ temporary_buffer<char> buf() {
+ return temporary_buffer<char>(msg.c_str(), msg.size());
+ }
+ };
+
+ std::vector<test_set> tests = {
+ { "14;name=value\r\n", true, "14", { {"name", "value"} } },
+ { "abcdef;name=value;name2=\"value2\"\r\n", true, "abcdef" },
+ { "1efg;name=value\r\n", false },
+ { "aa;tchars.^_`|123=t1!#$%&'*+-.~\r\n", true, "aa", { {"tchars.^_`|123", "t1!#$%&'*+-.~"} } },
+ { "1;quoted=\"hello world\";quoted-pair=\"\\a\\b\\cd\\\\ef\"\r\n", true, "1", { {"quoted", "hello world"}, {"quoted-pair", "abcd\\ef"} } },
+ { "2;bad-quoted-pair=\"abc\\\"\r\n", false },
+ { "3;quoted-pair-outside-quoted-string=\\q\\p\r\n", false },
+ { "4;whitespace-outside-quoted-string=quoted string\r\n", false },
+ { "5;quotation-mark-inside-quoted-string=\"quoted\"mark\"\r\n", false },
+ { "6; bad=space\r\n", false },
+ { "7;sole-name\r\n", true, "7", { { "sole-name", ""} } },
+ { "8;empty_value=\"\"\r\n", true, "8", { { "empty_value", ""} } },
+ { "0\r\n", true, "0" }
+ };
+
+ http_chunk_size_and_ext_parser parser;
+ for (auto& tset : tests) {
+ parser.init();
+ BOOST_REQUIRE(parser(tset.buf()).get0().has_value());
+ BOOST_REQUIRE_NE(parser.failed(), tset.parsable);
+ if (tset.parsable) {
+ BOOST_REQUIRE_EQUAL(parser.get_size(), std::move(tset.size));
+ auto exts = parser.get_parsed_extensions();
+ for (auto& ext : tset.extensions) {
+ BOOST_REQUIRE_EQUAL(exts[ext.first], ext.second);
+ }
+ }
+ }
+ return make_ready_future<>();
+}
+
+SEASTAR_TEST_CASE(test_trailer_headers_parsing) {
+ struct test_set {
+ sstring msg;
+ bool parsable;
+ sstring header_name = "";
+ sstring header_value = "";
+
+ temporary_buffer<char> buf() {
+ return temporary_buffer<char>(msg.c_str(), msg.size());
+ }
+ };
+
+ std::vector<test_set> tests = {
+ // the headers follow the same rules as in the request parser
+ { "Host: test\r\n\r\n", true, "Host", "test" },
+ { "Header: Field\r\n\r\n", true, "Header", "Field" },
+ { "Header: \r\n\r\n", true, "Header", "" },
+ { "Header: f i e l d \r\n\r\n", true, "Header", "f i e l d" },
+ { "Header: fiel\r\n d\r\n\r\n", true, "Header", "fiel d" },
+ { "tchars.^_`|123: printable!@#%^&*()obs_text\x80\x81\xff\r\n\r\n", true,
+ "tchars.^_`|123", "printable!@#%^&*()obs_text\x80\x81\xff" },
+ { "Header: Field\r\nHeader: Field2\r\n\r\n", true, "Header", "Field,Field2" },
+ { "Header : Field\r\n\r\n", false },
+ { "Header Field\r\n\r\n", false },
+ { "Header@: Field\r\n\r\n", false },
+ { "Header: fiel\r\nd \r\n\r\n", false },
+ { "\r\n", true }
+ };
+
+ http_chunk_trailer_parser parser;
+ for (auto& tset : tests) {
+ parser.init();
+ BOOST_REQUIRE(parser(tset.buf()).get0().has_value());
+ BOOST_REQUIRE_NE(parser.failed(), tset.parsable);
+ if (tset.parsable) {
+ auto heads = parser.get_parsed_headers();
+ BOOST_REQUIRE_EQUAL(heads[std::move(tset.header_name)], std::move(tset.header_value));
+ }
+ }
+ return make_ready_future<>();
+}