File: README.md

package info (click to toggle)
pdns-recursor 5.3.4-2
  • links: PTS, VCS
  • area: main
  • in suites: forky
  • size: 11,116 kB
  • sloc: cpp: 109,672; javascript: 20,651; python: 5,657; sh: 5,114; makefile: 782; ansic: 582; xml: 37
file content (227 lines) | stat: -rw-r--r-- 9,185 bytes parent folder | download | duplicates (3)
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
SETTINGS CODE
=============
This directory contains the code to generate both old-style as new style (YAML) settings code.

Inside this directory, there is a `rust` subdirectory that contains the Rust code and build files.
The Rust code uses CXX for bridging between C++ and Rust.
At the moment of writing, we only call Rust code from C++ and not vice versa.

Additionally, the Rust Serde crate (and specifically Serde-YAML) is used to generate code to handle YAML.

The entry point for code generation is `generate.py`, which uses `table.py` to produce C++, Rust and .rst files.
See `generate.py` for some details about the generation process.
This directory also contains a couple of `*-in.*` files which are included into files generated by the generation process.

From the C++ point of view, several namespaces are defined:

* `rust`: This namespace contains the classes defined by CXX.
Note that it is sometimes needed to explicitly name it `::rust`, as the `pdns` namespace also has a subnamespace called `rust`.
* `pdns::rust::settings::rec`: The classes and functions generated by CXX, callable from C++.
* `pdns::settings::rec`: The classes and functions of the settings code implemented in C++. This is mainly the code handling the conversion of old-style definitions to new-style (and vice versa).

Internally, the old-style settings are still used by the recursor code.
Rewriting the existing code to start using the new style settings directly would have made the PR introducing the new-style settings even bigger.
Therefore, we chose to keep the code *using* settings the same.
If a new-style settings YAML file is encountered, it will be parsed, validated and the resulting new-style settings will be used to set the old-style settings to the values found in the YAML file.
In the future, when old-style settings do not need to be supported any longer, the recursor itself can start to use the new style settings directly.

A `rec_control show-yaml [file]` command has been added to show the conversion of old-style settings to the new-style YAML.

This directory
--------------
* `cxxsettings-generated.cc`: generated code that implements the C++ part of the settings code.
* `cxxsettings-private.hh`: private interface used by C++ settings code internally.
* `cxxsettings.hh`: public interface of C++ code to be used by pdns_recursor and rec_control.
* `cxxsupport.cc`: hand written C++ code.
* `docs-new-preamble-in.rst`: non-generated part of new settings docs.
* `docs-old-preamble-in.rst`: non-generated part of old settings docs.
* `generate.py`: the Python script to produce C++, Rust and .rst files based on `table.py`.
* `rust`: the directory containing rust code.
* `rust-bridge-in.rs`: file included in the generated Rust code, placed inside the CXX bridge module.
* `rust-preamble-in.rs`: file included in the generated Rust code as a preamble.
* `table.py`: the definitions of all settings.

`rust` subdirectory
-------------------
* `Cargo.toml`: The definition of the Rust `settings` crate, including its dependencies.
* `build.rs`: `The custom build file used by CXX, see CXX docs.
* `cxx.h`: The generic types used by CXX generated code.
* `lib.rs.h`:  The project specific C++ types generated by CXX.
* `libsettings.a`: The actual static library produced by this crate.
* `src`: The actual rust code, `lib.rs` is generated, `bridge.rs` and `helpers.rs` are maintained manually.
* `target`: The `cargo` maintained Rust build directory.

The YAML settings are stored in a struct called (on the C++ side) `pdns::rust::settings::rec::Recursorsettings`.
This struct has a substruct for each section defined.
Each section has multiple typed variables.
Below we will tour some parts of the (generated) code.
An example settings file in YAML format:

```yaml
dnssec:
  log_bogus: true
incoming:
  listen:
  - 0.0.0.0:5301
  - '[::]:5301'
logging:
  common_errors: true
  disable_syslog: true
  loglevel: 6
recursor:
  daemon: false
  extended_resolution_errors: true
  socket_dir: /tmp/rec
  threads: 4
webservice:
  address: 127.0.0.1
  allow_from:
  - 0.0.0.0/0
  api_key: secret
  port: 8083
  webserver: true
```

The generated code
------------------
C++, Rust and documentation generating is done by the `generate.py` Python script using `table.py` as input.
After that, the C++ to Rust bridge code is generated by CXX.
Lets take a look at the `log_bogus` setting.
The source of its definition is in `table.py`:

```python
    {
        'name' : 'log_bogus',
        'section' : 'dnssec',
        'oldname' : 'dnssec-log-bogus',
        'type' : LType.Bool,
        'default' : 'false',
        'help' : 'Log DNSSEC bogus validations',
        'doc' : '''
Log every DNSSEC validation failure.
**Note**: This is not logged per-query but every time records are validated as Bogus.
 ''',
    },
```

The old-style documentation generated for this can be found in `../docs/settings.rst`:

```
.. _setting-dnssec-log-bogus:

``dnssec-log-bogus``
~~~~~~~~~~~~~~~~~~~~

-  Boolean
-  Default: no

- YAML setting: :ref:`setting-yaml-dnssec.log_bogus`

Log every DNSSEC validation failure.
**Note**: This is not logged per-query but every time records are validated as Bogus.
```

The new-style documentation generated for this can be found in `../docs/yamlsettings.rst`, its name includes the section and it lists a YAML default:

```
.. _setting-yaml-dnssec.log_bogus:

``dnssec.log_bogus``
^^^^^^^^^^^^^^^^^^^^

-  Boolean
-  Default: ``false``

- Old style setting: :ref:`setting-dnssec-log-bogus`

Log every DNSSEC validation failure.
**Note**: This is not logged per-query but every time records are validated as Bogus.
```

The C++ code generated from this entry can be found in `cxxsettings-generated.cc`.
The code to define the old-style settings, which is called by the recursor very early after startup and is a replacement for the hand-written code that defines all settings in the old recursor code.
Using generated code and generated docs makes sure the docs and the actual implementation are consistent.
Something which was not true for the old code in all cases.

```cpp
void pdns::settings::rec::defineOldStyleSettings()
  ...
  ::arg().setSwitch("dnssec-log-bogus", "Log DNSSEC bogus validations") = "no";
  ...
```

There is also code generated to assign the current value of old-style `log_bogus` value to the right field in a `Recursorsettings` struct.
This code is used to convert the currently active settings to a YAML file (`pdns_recursor --config=diff`):
```cpp
void pdns::settings::rec::oldStyleSettingsToBridgeStruct(Recursorsettings& settings)
  ...
  settings.dnssec.log_bogus = arg().mustDo("dnssec-log-bogus");
  ...
```

Plus code to set the old-style setting, given a new-style struct:

```cpp
void pdns::settings::rec::bridgeStructToOldStyleSettings(const Recursorsettings& settings)
  ...
  ::arg().set("dnssec-log-bogus") = to_arg(settings.dnssec.log_bogus);
  ...
```

Lastly, there is code to support converting a old-style settings to a new-style struct found in `pdns::settings::rec::oldKVToBridgeStruct()`.
This code is used to convert old style settings files to YAML (used by `rec_control show-yaml`) and to generate a yaml file with all the defaults values (used by `pdns_recursor --config=default`).
The functions implementing that are `pdns::settings::rec::oldStyleSettingsFileToYaml` and `std::string pdns::settings::rec::defaultsToYaml()`, found in `cxxsupport.cc`.

The Rust code generated by `generate.py` can be found in `rust/src/lib.rs`.
It contains these snippets that define the `log_bogus` field in the section struct `Dnssec` and the `dnssec` field in the `Recursorsettings` struct:

```rust
pub struct Dnssec {
   ...
   #[serde(default, skip_serializing_if = "crate::is_default")]
   log_bogus: bool,
   ...
}
pub struct Recursorsettings {
   ...
   #[serde(default, skip_serializing_if = "crate::is_default")]
   dnssec: Dnssec,
   ...
}
```
The `rust/src/lib.rs` file also contains the generated code to handle Serde defaults.
More details on this can be found in `generate.py`.

The C++ version of the `Recursorsettings` struct and its substructs can be found in the CXX generated `rust/lib.rs.h` file:

```cpp
struct Recursorsettings final {
   ...
   ::pdns::rust::settings::rec::Dnssec dnssec;
   ...
   };
   ...
struct Dnssec final {
  ...
  bool log_bogus;
  ...
};
```

The Rust functions callable from C++ are listed in `rust-bridge-in.rs` which gets included into `rust/src/lib.rs` by `generate.py`
An example is the function

```rust
  fn parse_yaml_string(str: &String) -> Result<Recursorsettings>;
```

Which parses YAML and produces a struct with all the settings.
Settings that are not mentioned in the YAML string will have their default value.

`rust/lib.rs.h` contains the corresponding C++ prototype, defined in the `pdns::rust::settings::rec` namespace:

```cpp
::pdns::rust::settings::rec::Recursorsettings parse_yaml_string(::rust::String const &str);
```

The C++ function `pdns::settings::rec::readYamlSettings()` defined in `cxxsupport.cc` and called in `../rec-main.cc` calls the Rust function `pdns::rust::settings::rec::parse_yaml_string()` to do the actual YAML parsing.