Kea 2.7.6
option_custom.h
Go to the documentation of this file.
1// Copyright (C) 2012-2024 Internet Systems Consortium, Inc. ("ISC")
2//
3// This Source Code Form is subject to the terms of the Mozilla Public
4// License, v. 2.0. If a copy of the MPL was not distributed with this
5// file, You can obtain one at http://mozilla.org/MPL/2.0/.
6
7#ifndef OPTION_CUSTOM_H
8#define OPTION_CUSTOM_H
9
10#include <asiolink/io_address.h>
11#include <dhcp/option.h>
14
15namespace isc {
16namespace dhcp {
17
31class OptionCustom : public Option {
32public:
33
44
60 OptionCustom(const OptionDefinition& def, Universe u, const OptionBuffer& data);
61
82
84 virtual OptionPtr clone() const;
85
90 void addArrayDataField(const asiolink::IOAddress& address);
91
95 void addArrayDataField(const bool value);
96
101 template<typename T>
102 void addArrayDataField(const T value) {
103 checkArrayType();
104 OptionDataType data_type = definition_.getType();
105 // Handle record last field.
106 if (data_type == OPT_RECORD_TYPE) {
107 data_type = definition_.getRecordFields().back();
108 }
109 if (OptionDataTypeTraits<T>::type != data_type) {
111 "specified data type " << data_type << " does not"
112 " match the data type in an option definition");
113 }
114
115 OptionBuffer buf;
117 buffers_.push_back(buf);
118 }
119
123 void addArrayDataField(const std::string& value);
124
128 void addArrayDataField(const OpaqueDataTuple& value);
129
134 void addArrayDataField(const PrefixLen& prefix_len,
135 const asiolink::IOAddress& prefix);
136
141 void addArrayDataField(const PSIDLen& psid_len, const PSID& psid);
142
146 uint32_t getDataFieldsNum() const { return (buffers_.size()); }
147
154 asiolink::IOAddress readAddress(const uint32_t index = 0) const;
155
163 void writeAddress(const asiolink::IOAddress& address,
164 const uint32_t index = 0);
165
172 const OptionBuffer& readBinary(const uint32_t index = 0) const;
173
178 void writeBinary(const OptionBuffer& buf, const uint32_t index = 0);
179
186 std::string readTuple(const uint32_t index = 0) const;
187
194 void readTuple(OpaqueDataTuple& tuple, const uint32_t index = 0) const;
195
200 void writeTuple(const std::string& value, const uint32_t index = 0);
201
206 void writeTuple(const OpaqueDataTuple& value, const uint32_t index = 0);
207
214 bool readBoolean(const uint32_t index = 0) const;
215
222 void writeBoolean(const bool value, const uint32_t index = 0);
223
232 std::string readFqdn(const uint32_t index = 0) const;
233
240 void writeFqdn(const std::string& fqdn, const uint32_t index = 0);
241
250 template<typename T>
251 T readInteger(const uint32_t index = 0) const {
252 // Check that the index is not out of range.
253 checkIndex(index);
254 // Check that T points to a valid integer type and this type
255 // is consistent with an option definition.
256 checkDataType<T>(index);
257 // When we created the buffer we have checked that it has a
258 // valid size so this condition here should be always fulfilled.
259 isc_throw_assert(buffers_[index].size() == OptionDataTypeTraits<T>::len);
260 // Read an integer value.
261 return (OptionDataTypeUtil::readInt<T>(buffers_[index]));
262 }
263
272 template<typename T>
273 void writeInteger(const T value, const uint32_t index = 0) {
274 // Check that the index is not out of range.
275 checkIndex(index);
276 // Check that T points to a valid integer type and this type
277 // is consistent with an option definition.
278 checkDataType<T>(index);
279 // Get some temporary buffer.
280 OptionBuffer buf;
281 // Try to write to the buffer.
283 // If successful, replace the old buffer with new one.
284 std::swap(buffers_[index], buf);
285 }
286
293 PrefixTuple readPrefix(const uint32_t index = 0) const;
294
302 void writePrefix(const PrefixLen& prefix_len,
303 const asiolink::IOAddress& prefix,
304 const uint32_t index = 0);
305
312 PSIDTuple readPsid(const uint32_t index = 0) const;
313
323 void writePsid(const PSIDLen& psid_len, const PSID& psid,
324 const uint32_t index = 0);
325
332 std::string readString(const uint32_t index = 0) const;
333
338 void writeString(const std::string& text,
339 const uint32_t index = 0);
340
345 virtual void pack(isc::util::OutputBuffer& buf, bool check = true) const;
346
351 virtual void unpack(OptionBufferConstIter begin,
353
359 virtual std::string toText(int indent = 0) const;
360
365 virtual uint16_t len() const;
366
373 void initialize(const OptionBufferConstIter first,
374 const OptionBufferConstIter last);
375
376private:
377
385 inline void checkArrayType() const {
386 if (!definition_.getArrayType()) {
387 isc_throw(InvalidOperation, "failed to add new array entry to an"
388 << " option. The option is not an array.");
389 }
390 }
391
402 template<typename T>
403 // cppcheck-suppress unusedPrivateFunction
404 void checkDataType(const uint32_t index) const;
405
411 void checkIndex(const uint32_t index) const;
412
417 void createBuffer(OptionBuffer& buffer,
418 const OptionDataType data_type) const;
419
421 void createBuffers();
422
432 size_t bufferLength(const OptionDataType data_type, bool in_array,
433 OptionBuffer::const_iterator begin,
434 OptionBuffer::const_iterator end) const;
435
439 void createBuffers(const OptionBuffer& data_buf);
440
447 std::string dataFieldToText(const OptionDataType data_type,
448 const uint32_t index) const;
449
453 using Option::setData;
454
456 OptionDefinition definition_;
457
461 std::vector<OptionBuffer> buffers_;
462};
463
465typedef boost::shared_ptr<OptionCustom> OptionCustomPtr;
466
467template<typename T>
468void
469OptionCustom::checkDataType(const uint32_t index) const {
470 // Check that the requested return type is a supported integer.
472 isc_throw(isc::dhcp::InvalidDataType, "specified data type"
473 " is not a supported integer type.");
474 }
475
476 // Get the option definition type.
477 OptionDataType data_type = definition_.getType();
478 if (data_type == OPT_RECORD_TYPE) {
479 const OptionDefinition::RecordFieldsCollection& record_fields =
480 definition_.getRecordFields();
481 if (definition_.getArrayType()) {
482 // If the array flag is set the last record field is an array.
483 if (index < record_fields.size()) {
484 // Get the data type to be returned.
485 data_type = record_fields[index];
486 } else {
487 // Get the data type to be returned from the last record field.
488 data_type = record_fields.back();
489 }
490 } else {
491 // When we initialized buffers we have already checked that
492 // the number of these buffers is equal to number of option
493 // fields in the record so the condition below should be met.
494 isc_throw_assert(index < record_fields.size());
495 // Get the data type to be returned.
496 data_type = record_fields[index];
497 }
498 }
499
500 if (OptionDataTypeTraits<T>::type != data_type) {
502 "specified data type " << data_type << " does not"
503 " match the data type in an option definition for field"
504 " index " << index);
505 }
506}
507} // namespace isc::dhcp
508} // namespace isc
509
510#endif // OPTION_CUSTOM_H
A generic exception that is thrown if a function is called in a prohibited way.
Exception to be thrown when invalid type specified as template parameter.
Represents a single instance of the opaque data preceded by length.
Option with defined data fields represented as buffers that can be accessed using data field index.
std::string readString(const uint32_t index=0) const
Read a buffer as string value.
void addArrayDataField(const T value)
Create new buffer and store integer value in it.
bool readBoolean(const uint32_t index=0) const
Read a buffer as boolean value.
virtual uint16_t len() const
Returns length of the complete option (data length + DHCPv4/DHCPv6 option header)
std::string readTuple(const uint32_t index=0) const
Read a buffer as length and string tuple.
void writeFqdn(const std::string &fqdn, const uint32_t index=0)
Write an FQDN into a buffer.
std::string readFqdn(const uint32_t index=0) const
Read a buffer as FQDN.
void writePrefix(const PrefixLen &prefix_len, const asiolink::IOAddress &prefix, const uint32_t index=0)
Write prefix length and value into a buffer.
virtual void unpack(OptionBufferConstIter begin, OptionBufferConstIter end)
Parses received buffer.
void writeAddress(const asiolink::IOAddress &address, const uint32_t index=0)
Write an IP address into a buffer.
virtual void pack(isc::util::OutputBuffer &buf, bool check=true) const
Writes DHCP option in a wire format to a buffer.
void initialize(const OptionBufferConstIter first, const OptionBufferConstIter last)
Sets content of this option from buffer.
const OptionBuffer & readBinary(const uint32_t index=0) const
Read a buffer as binary data.
PrefixTuple readPrefix(const uint32_t index=0) const
Read a buffer as variable length prefix.
void writePsid(const PSIDLen &psid_len, const PSID &psid, const uint32_t index=0)
Write PSID length / value into a buffer.
void writeBoolean(const bool value, const uint32_t index=0)
Write a boolean value into a buffer.
asiolink::IOAddress readAddress(const uint32_t index=0) const
Read a buffer as IP address.
PSIDTuple readPsid(const uint32_t index=0) const
Read a buffer as a PSID length / value tuple.
void writeString(const std::string &text, const uint32_t index=0)
Write a string value into a buffer.
T readInteger(const uint32_t index=0) const
Read a buffer as integer value.
void writeBinary(const OptionBuffer &buf, const uint32_t index=0)
Write binary data into a buffer.
void addArrayDataField(const asiolink::IOAddress &address)
Create new buffer and set its value as an IP address.
virtual OptionPtr clone() const
Copies this option and returns a pointer to the copy.
void writeTuple(const std::string &value, const uint32_t index=0)
Write a length and string tuple into a buffer.
virtual std::string toText(int indent=0) const
Returns string representation of the option.
OptionCustom(const OptionDefinition &def, Universe u)
Constructor, used for options to be sent.
uint32_t getDataFieldsNum() const
Return a number of the data fields.
void writeInteger(const T value, const uint32_t index=0)
Write an integer value into a buffer.
static void writeInt(const T value, std::vector< uint8_t > &buf)
Append integer or unsigned integer value to a buffer.
static T readInt(const std::vector< uint8_t > &buf)
Read integer value from a buffer.
Base class representing a DHCP option definition.
OptionDataType getType() const
Return option data type.
std::vector< OptionDataType > RecordFieldsCollection
List of fields within the record.
const RecordFieldsCollection & getRecordFields() const
Return list of record fields.
bool getArrayType() const
Return array type indicator.
Universe
defines option universe DHCPv4 or DHCPv6
Definition option.h:90
void setData(InputIterator first, InputIterator last)
Sets content of this option from buffer.
Definition option.h:434
void check() const
A protected method used for option correctness.
Definition option.cc:90
Encapsulates PSID length.
Encapsulates PSID value.
Encapsulates prefix length.
The OutputBuffer class is a buffer abstraction for manipulating mutable data.
Definition buffer.h:343
#define isc_throw(type, stream)
A shortcut macro to insert known values into exception arguments.
#define isc_throw_assert(expr)
Replacement for assert() that throws if the expression is false.
Definition isc_assert.h:18
std::pair< PSIDLen, PSID > PSIDTuple
Defines a pair of PSID length / value.
boost::shared_ptr< OptionCustom > OptionCustomPtr
A pointer to the OptionCustom object.
OptionBuffer::const_iterator OptionBufferConstIter
const_iterator for walking over OptionBuffer
Definition option.h:30
OptionDataType
Data types of DHCP option fields.
std::pair< PrefixLen, asiolink::IOAddress > PrefixTuple
Defines a pair of prefix length / value.
std::vector< uint8_t > OptionBuffer
buffer types used in DHCP code.
Definition option.h:24
boost::shared_ptr< Option > OptionPtr
Definition option.h:37
Defines the logger used by the top-level component of kea-lfc.
Trait class for data types supported in DHCP option definitions.