-
Notifications
You must be signed in to change notification settings - Fork 191
Expand file tree
/
Copy pathhttp_method.hpp
More file actions
262 lines (218 loc) · 9.16 KB
/
Copy pathhttp_method.hpp
File metadata and controls
262 lines (218 loc) · 9.16 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
/*
This file is part of libhttpserver
Copyright (C) 2011-2019 Sebastiano Merlino
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
USA
*/
#if !defined (_HTTPSERVER_HPP_INSIDE_) && !defined (HTTPSERVER_COMPILATION)
#error "Only <httpserver.hpp> or <httpserverpp> can be included directly."
#endif
#ifndef SRC_HTTPSERVER_HTTP_METHOD_HPP_
#define SRC_HTTPSERVER_HTTP_METHOD_HPP_
#include <array>
#include <cstddef>
#include <cstdint>
#include <string_view>
#include <type_traits>
namespace httpserver {
// Strongly-typed HTTP method primitive consumed by http_resource, the
// route table, and lambda registration. The identifier `del` (rather
// than `delete`) avoids the C++ keyword; the wire-protocol token
// returned by to_string() is "DELETE".
//
// `count_` is a sentinel and must remain the last enumerator. Any new
// method goes immediately before it; to_string()'s switch must also be
// updated. The 32-bit underlying storage of method_set leaves 23 bits
// of growth headroom past the 9 standard methods (PRD-REQ-REQ-003,
// DR-006).
enum class http_method : std::uint8_t {
get,
head,
post,
put,
del, // wire token "DELETE"
connect,
options,
trace,
patch,
count_ // sentinel; must remain last
};
namespace detail {
// Bit position for an http_method enumerator. Defined here so member
// functions and free operators can share one definition. Out-of-range
// inputs (>= 32) are masked out by the caller; this helper is total.
constexpr std::uint32_t method_bit(http_method m) noexcept {
return std::uint32_t{1} << static_cast<std::uint8_t>(m);
}
// All-valid-methods mask: bits 0 .. count_-1 set, the rest cleared.
constexpr std::uint32_t valid_method_mask() noexcept {
return (std::uint32_t{1}
<< static_cast<std::uint8_t>(http_method::count_)) - 1u;
}
} // namespace detail
// Fixed-size set of allowed HTTP methods (one bit per http_method
// enumerator). Aggregate so it stays standard layout / trivially
// copyable; brace-init with {bits} is fine, and default-init gives an
// empty set. Comparison is defaulted (constexpr noexcept).
struct method_set {
std::uint32_t bits = 0;
constexpr bool contains(http_method m) const noexcept {
return (bits & detail::method_bit(m)) != 0u;
}
constexpr method_set& set(http_method m) noexcept {
bits |= detail::method_bit(m);
return *this;
}
constexpr method_set& clear(http_method m) noexcept {
bits &= ~detail::method_bit(m);
return *this;
}
// set_all() and clear_all() operate over the valid-method window
// (bits 0 .. count_-1); bits beyond count_ stay zero so complement
// round-trips cleanly.
constexpr method_set& set_all() noexcept {
bits = detail::valid_method_mask();
return *this;
}
constexpr method_set& clear_all() noexcept {
bits = 0u;
return *this;
}
// TASK-027: convenience predicate for the route-table writer paths
// (on_methods_, route(method_set,...)). True iff no method bits are
// set. Constexpr noexcept so it works in the same constant-context
// gates as the rest of the method_set surface.
constexpr bool empty() const noexcept {
return bits == 0u;
}
friend constexpr bool operator==(method_set, method_set) noexcept = default;
};
// to_string returns the uppercase RFC 9110 wire token for use in logs
// and the 405 Allow: header. Total over the 9 declared enumerators;
// any other underlying value (only producible via static_cast) returns
// an empty view rather than crashing — keeps logging robust against
// stale enum values.
constexpr std::string_view to_string(http_method m) noexcept {
// Indexed by the underlying enum value (0..count_-1). Out-of-range
// values (only producible via static_cast) return an empty view.
// The order MUST stay aligned with the http_method enum declaration
// (TASK-021); a static_assert on count_ catches drift.
constexpr std::array<std::string_view, static_cast<std::size_t>(http_method::count_)> names{
std::string_view{"GET"},
std::string_view{"HEAD"},
std::string_view{"POST"},
std::string_view{"PUT"},
std::string_view{"DELETE"},
std::string_view{"CONNECT"},
std::string_view{"OPTIONS"},
std::string_view{"TRACE"},
std::string_view{"PATCH"},
};
const auto idx = static_cast<std::size_t>(m);
if (idx >= names.size()) return std::string_view{};
return names[idx];
}
// Bitwise composition. Operators on http_method yield a method_set so
// `get | post` is a two-method set ready to feed into route_entry.
// All operators are constexpr noexcept — usable in compile-time
// context (the "consteval-friendly" requirement) AND at runtime, which
// the route-table writer path needs.
constexpr method_set operator|(http_method a, http_method b) noexcept {
return method_set{detail::method_bit(a) | detail::method_bit(b)};
}
constexpr method_set operator&(http_method a, http_method b) noexcept {
return method_set{detail::method_bit(a) & detail::method_bit(b)};
}
constexpr method_set operator^(http_method a, http_method b) noexcept {
return method_set{detail::method_bit(a) ^ detail::method_bit(b)};
}
// ~http_method == "every valid method except this one" (bounded to the
// count_ window).
constexpr method_set operator~(http_method m) noexcept {
return method_set{detail::valid_method_mask() & ~detail::method_bit(m)};
}
constexpr method_set operator|(method_set a, method_set b) noexcept {
return method_set{a.bits | b.bits};
}
constexpr method_set operator&(method_set a, method_set b) noexcept {
return method_set{a.bits & b.bits};
}
constexpr method_set operator^(method_set a, method_set b) noexcept {
return method_set{a.bits ^ b.bits};
}
// ~method_set is also bounded to the valid-method window so
// `~method_set{}.set_all() == method_set{}` holds — i.e. complement is
// an involution within the 9-bit window. Without the masking, unused
// upper bits would leak in and break round-tripping.
constexpr method_set operator~(method_set s) noexcept {
return method_set{detail::valid_method_mask() & ~s.bits};
}
// Mixed (method_set, http_method) overloads — convenience for the
// common "set | method" composition.
constexpr method_set operator|(method_set s, http_method m) noexcept {
return method_set{s.bits | detail::method_bit(m)};
}
constexpr method_set operator|(http_method m, method_set s) noexcept {
return s | m;
}
constexpr method_set operator&(method_set s, http_method m) noexcept {
return method_set{s.bits & detail::method_bit(m)};
}
constexpr method_set operator&(http_method m, method_set s) noexcept {
return s & m;
}
constexpr method_set operator^(method_set s, http_method m) noexcept {
return method_set{s.bits ^ detail::method_bit(m)};
}
constexpr method_set operator^(http_method m, method_set s) noexcept {
return s ^ m;
}
// Compound assignment on method_set (free functions to match the
// non-member binary operators above).
constexpr method_set& operator|=(method_set& s, method_set rhs) noexcept {
s.bits |= rhs.bits;
return s;
}
constexpr method_set& operator&=(method_set& s, method_set rhs) noexcept {
s.bits &= rhs.bits;
return s;
}
constexpr method_set& operator^=(method_set& s, method_set rhs) noexcept {
s.bits ^= rhs.bits;
return s;
}
constexpr method_set& operator|=(method_set& s, http_method m) noexcept {
s.bits |= detail::method_bit(m);
return s;
}
constexpr method_set& operator&=(method_set& s, http_method m) noexcept {
s.bits &= detail::method_bit(m);
return s;
}
constexpr method_set& operator^=(method_set& s, http_method m) noexcept {
s.bits ^= detail::method_bit(m);
return s;
}
// Layout / width invariants — pinned once at namespace scope so every
// TU including this header gets the protection. Placed AFTER the
// method_set definition so is_standard_layout_v / sizeof are well-formed.
static_assert(static_cast<std::uint8_t>(http_method::count_) <= 32,
"http_method::count_ must fit in method_set's 32-bit bitmask");
static_assert(std::is_standard_layout_v<method_set>,
"method_set must be standard layout");
static_assert(std::is_trivially_copyable_v<method_set>,
"method_set must be trivially copyable");
static_assert(sizeof(method_set) == sizeof(std::uint32_t),
"method_set must be exactly the size of its underlying uint32_t");
} // namespace httpserver
#endif // SRC_HTTPSERVER_HTTP_METHOD_HPP_