Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor(dslx): start making functions stateless #1379

Merged
merged 11 commits into from
Oct 25, 2023
Merged

Conversation

bassosimone
Copy link
Contributor

@bassosimone bassosimone commented Oct 25, 2023

This diff is pretty messy unless you use git diff -w, in which case you see changes are actually minimal.

With this diff, we implement roughly 50% of ooni/probe#2612. I have another couple of diffs ready to finish off this issue's work, but committing all of them together would create too much noise.

We want to use the DSL inside the oohelperd. We don't care about
collecting observations in the oohelperd. So, the plan is that of
abstracting the ConnPool, renaming it Runtime, and giving it the
power to create abstract Traces.

The oohelperd will use a MinimalRuntime (sketched out by this
commit) that will not collect any observation.

Measuring code, instead, will use a MeasurexRuntime that will
collect observations.

This commit is just the first step. We rename and introduce the
MinimalRuntime. No significant functional changes so far.
This diff builds on the previous diff and uses an abstract trace
inside of the dslx package. By using an abstract trace, we can
choose between using:

- a runtime that collects observations, based on measurexlite; and

- a minimal runtime that does not collect observations.

To make the trace abstract, we need to modify measurexlite's trace
such that it can be used as an interface.

In turn, this means we need to update the throttling package
such that it uses an abstract trace definition. Strictly speaking,
we could have avoided introducing this abstraction, but it seems
better to also use an abstract trace there, as it allows for
improving the decoupling with measurexlite.
Currently, we pass these fields to each DSL function. However, if
we want to load the functions from JSON, which is something we have
experimented with in the richer input context, we can't do this.

These fields should instead belong to the runtime. A subsequent diff
will modify the DSL functions to take them from the runtime.
This diff builds upon the previous diff to use the Runtime to get
the logger, ID generator, and zero time.

By doing this, we make most structures that DSL functions takes
as input or emit in output serializable and deserializable.
Rather than using custom fields for testing, we can configure in the
runtime a custom model.MeasuringNetwork.

We're not doing this just to simplify the codebase, rather the underlying
intent here is making sure we don't need to keep much state in each
function, so we can refactor them to be pure functions wrapped by an
adapter that produces the desired type.

In turn, by doing that, we will be able to factor complexity around
invoking functions and parsing their results.

In turn, by doing that, we will be able to modify the signature of the
functions and do the following:

1. allow the DSL model to include stages that take in input a Maybe
value rather than a value, so we can observe failures more easily than
we do now and we can write inline code to save into test keys;

2. allow the DSL model to much more easily be refactored to use
channels, which in turn enables us to compose operations more naturally
and increase the amount of overlapping (think, e.g., how this enables
the possibility of waiting additional time for a DNS-over-UDP resolver
to wait for late/duplicate replies).
This diff modifies dslx functions to always use the MeasuringNetwork
for testing rather than using specific func fields.

By doing this, we open up the possibility of simplifying the state of
each func, with the ultimate goal of making them pure functions.

By making them pure functions, we make the code more manageable and
easy to modify, which opens up for additional refactorings.
Introduce an adapter type that converts a pure function into a Func
and start using it whenever it's easy, rather than rolling out structs
that implement the Func type when we actually don't have state.

This change is an improvement because we're creating the necessary
conditions from moving complexity out of the functions that actually
do something, which should be simpler, and adapters, which should
contain the same, equal logic for creating pipelines.

The ultimate goal here is to be able to have stages that accept
a Maybe[A] as input, to be able to add inline evaluators that set
test keys. I want to reach this goal by making the necessary
transformation at the adapters level rather than changing each
of the Func prototype at once.
This diff reduces the state kept by the tlsHandshakeFunc struct
so that we can apply a transformation similar to the one we applied
for TCPConnect() and implement TLSHandshake() using a pure func.

The overall objective is that of factoring away completixity to
enable manipulating this code more easily.

While there, let's note that the changes applied here mean that we
can reuse this code for configuring tls.Config for the QUICHandshake.
This diff takes advantage of the fact that now the TLSHandshakeOption
are independent of the tlsHandshakeFunc structure, so we can use the
same options for configuring the QUIC handshake.
Now that we have removed the state from the underlying implementation
of TLSHandshake and QUICHandshake, we can rewrite the code to use
pure functions as opposed to using unneeded state.
Conflicts:
	internal/dslx/dns.go
	internal/dslx/dns_test.go
	internal/dslx/quic.go
	internal/dslx/quic_test.go
	internal/dslx/tcp.go
	internal/dslx/tcp_test.go
	internal/dslx/tls.go
@bassosimone bassosimone changed the title Issue/2545 small 2 refactor(dslx): start making functions stateless Oct 25, 2023
@bassosimone bassosimone marked this pull request as ready for review October 25, 2023 07:45
@bassosimone bassosimone requested a review from hellais as a code owner October 25, 2023 07:45
@bassosimone bassosimone merged commit 600c7f6 into master Oct 25, 2023
8 checks passed
@bassosimone bassosimone deleted the issue/2545-small-2 branch October 25, 2023 07:51
Murphy-OrangeMud pushed a commit to Murphy-OrangeMud/probe-cli that referenced this pull request Feb 13, 2024
This diff is pretty messy unless you use `git diff -w`, in which case
you see changes are actually minimal.

With this diff, we implement roughly 50% of
ooni/probe#2612. I have another couple of
diffs ready to finish off this issue's work, but committing all of them
together would create too much noise.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

1 participant