Skip to content

Latest commit

 

History

History
66 lines (33 loc) · 5.18 KB

Implementation Guidelines.md

File metadata and controls

66 lines (33 loc) · 5.18 KB

Guidelines to implementing tests with Zonemaster

Zonemaster is a Perl framework meant to aid the implementation of DNS tests. The design of the framework is based primarily on IIS's experiences with the DNSCheck tool and the top-level-domain Pre-Delegation Testing platform. This document aims to present the rationales for the various design decisions in Zonemaster.

Goals of the system as a whole

All design decisions are intended to contribute toward one or more of a number of overall goals. These are, in rough order of descending importance:

  1. Correctness

    First and foremost, the results produced by tests must be correct. By correct here we mean that they accurately implement the specification for a given test, and that they can be verified to do so by a competent observer.

  2. Predictability

    Given an identical view of the outside world, the test must always give the exact same result. There must be nothing random or arbitrary in the implementation (or the specification).

  3. Transparency

    We should be able to tell after the fact why a test produced the result it did. It is acceptable for this to require re-running a test with settings that record more detailed (and thus voluminous) information, but in the extreme case it should be possible to go back and look at the details of every single incoming DNS packet in order to figure out why something strange happened.

  4. Modifiability

    We should be able to modify the test's view of the outside world. The must common use for this is to test zones that have not yet been delegated from their intended parents.

Looking at Zonemaster with the goals in mind

In the Zonemaster framework, tests are implemented in plugin modules. These modules must have a method called all, which takes a zone object as its single argument. This object represents the entire outside world for the tests. Test code must never communicate with the outside world (like for example nameservers) except via the methods in the zone object.

The zone object is the world

By forcing all communication to pass through a single chokepoint, we serve all of the goals listed above. For example, the query_one method makes sure to always send queries to the same child-side server (the first one in lexicographic order by stringified name). The query methods also have the ability to transparently store incoming packets for future inspection, and it will have the ability to answer queries with pre-loaded information instead of sending them out into the world (which makes it vastly easier to write unit tests for the testing code). And by having all this functionality hidden away in another object, the test code itself can be written in a more straightforward way, which in turn makes it easier to determine if it is correct.

Zonemaster packet and record objects have helpers

The less code there is, the easier it is to determine if it actually does what it should. When writing DNS tests, there are certain actions that come up often. In order to reduce the amount of code implementing the tests themselves, we extract such actions into helper methods in the objects representing DNS packets and (possibly in the future) resource records.

The intention is that as often occuring patterns are found, they will be extracted out into surrounding objects. As a rule of thumb, I'd suggest that once a certain structure has been needed in three or more places, extracting it out should be considered.

Implementation guidelines

Returning information from tests, the theory

There are basically two levels at which a user is interested in the result of a test: only the plain results, or the results plus intermediate and debugging information created during the testing process. They way this separation is intended to work in Zonemaster is by a test implementation creating log entry objects for all information that might be interesting, but only return a list of the ones representing actual test results to the caller. The other ones will be kept track of by the framework, and a user can ask to get them if and when they so desire. So a user who is only interested in the results will run the test(s) and get the results as a list of log entry objects as the return value. A user who has more complex needs can ask the framework to also get the rest of the log entries.

Test Policy

It is recommended, although not required, that each test case be implemented as its own method. What is required is that a test module check with the global configuration that a certain test case should be executed before it does so.

Code formatting

The top level of the Zonemaster git repository contains .perltidyrc, a config file for Perl::Tidy. Please use it before you push, make a pull request or otherwise send code outside of your own repository.

Code style

The top level of the Zonemaster git repository also contains .perlcriticrc, a config file for Perl::Critic. Strive to make your code free from warnings on at least levels 4 and 5 before you send it out into the world.

Exposing test metadata

Returning information from tests, the practice

(to be continued)