Page MenuHomeFreeBSD

Add the "missing" WireGuard rc.d script
Needs ReviewPublic

Authored by crest_freebsd_rlwinm.de on Aug 4 2023, 10:44 AM.
Referenced Files
Unknown Object (File)
Fri, Jan 3, 6:46 PM
Unknown Object (File)
Fri, Jan 3, 6:42 PM
Unknown Object (File)
Dec 7 2024, 9:58 AM
Unknown Object (File)
Nov 24 2024, 10:36 PM
Unknown Object (File)
Nov 23 2024, 10:13 AM
Unknown Object (File)
Nov 22 2024, 1:40 PM
Unknown Object (File)
Nov 21 2024, 2:19 PM
Unknown Object (File)
Nov 20 2024, 2:21 AM

Details

Reviewers
jhb
bapt
olivier
kevans
Group Reviewers
manpages
Summary

This rc.d script reimplements most features of wg-quick(8) in FreeBSD
/bin/sh instead of bash.

The following features have not been reimplemented:

  • The daemon syncing a routing table the AllowedIPs settings.
  • Saving the active configuration back to a config file.

The PreUp/PostUp/PreDown/PostDown hooks are implemented, but the hooks
are executed in sh instead of bash. Additionally the successful execution
is tracked to avoid (re-)execution of hooks because it's expected that users will rerun rc.d scripts.

Since the WireGuard rc.d script is supposed to make using WireGuard on FreeBSD as easy as possible it handles the creation and destruction of WireGuard interfaces instead of forcing users to configure the netif script to clone and rename WireGuard interfaces I added one feature not present in wg-quick(8): the Sticky setting. Sticky interfaces are only deconfigured (IP addresses removed, link brought down, configuration replaced with the empty configuration), but not destroyed by service wireguard stop $ifn. To avoid having to add a FreeBSD specific setting to the configuration files the rc.d script also looks at the sticky bit of interface configuration files.

The rc.d script uses per interface lock files (implemented using lockf) to protect against race conditions (e.g. restarting an interface via devd and ifstated at the same time), because otherwise users would have to implement similar locking in their {Pre,Post}{Up,Down} hooks.

Test Plan
  • Create /etc/wireguard using install -d -m 750 -o root -g wheel /etc/wireguard
  • Create a test interface configuration e.g. ${EDITOR:-vi} /etc/wireguard/wg-test.conf
  • Start the rc.d script: service wireguard start.
  • Restart just one interface: service wireguard restart wg-test
  • Stop the rc.d script: service wireguard stop.

The proposed test plan covers only the basic operations since there is no way to foresee what users will put in their hooks.

Diff Detail

Repository
rG FreeBSD src repository
Lint
Lint Skipped
Unit
Tests Skipped

Event Timeline

first pass…

some of the formatting looks really off. Especially the case statements.
Please make sure you're using tabs everywhere.

I'll have more comments when I'm less tired, but for now we'll need entries in /etc/defaults/rc.conf and in rc.conf(5)

libexec/rc/rc.d/wireguard
57

maybe this function should go into /etc/network.subr

The interface name restriction function is a judgment call restricting users from creating problematic (for shell scripts) interface names. Applying the same restrictions to existing services like netif and routing could break (partly) working configurations.

libexec/rc/rc.d/wireguard
57

This function is a judgment call restricting users from creating problematic (for shell scripts) interface names. Applying the same restrictions to existing services like netif and routing could break (partly) working configurations.

This is my first review here, I hope to not be stepping on any toes. If I have, please correct me so I might do better next time. Overall your wireguard startup mechanism looks good and my suggestions are strictly cosmetic in nature. I would not be at all disappointed if this patch were committed unaltered.

A general comment is that there are several instances similar to:

	if ! exists; then
		warn "Missing WireGuard interface $INTERFACE to bring down."
	elif ! in_wg; then
		warn "Refusing to bring down non-WireGuard interface $INTERFACE."

I wonder if the code wouldn't be overall more easily read if those lines were factored out into a small helper function so we only every saw validate_if $INTERFACE <caller function name>?

libexec/rc/rc.d/wireguard
3

I believe that we've discontinued the use of these rcs/cvs tags in our shell scripts.

233

Perhaps lose the 6085 string in the middle of this line of hash marks?

527

I don't believe that the enter is needed if state was already $1, maybe move this line above the 'fi' on the preceding line?

687

The inconsistent use of white-space in the next dozen or so lines is a bit distracting.

No manual page to review, yet manpages is a group reviewer. Did a file get accidentally left out?

No manual page to review, yet manpages is a group reviewer. Did a file get accidentally left out?

There is no change to man pages yet, but the new rc.conf(5) variables have to be documented (e.g. duplicating what service wireguard describe would print) as man page.

libexec/rc/rc.d/wireguard
3

Correct. This line got overcome by events.

libexec/rc/rc.d/wireguard
233

Will do.

527

It doesn't hurt as it's a cheap function to call. It can be argued both ways, but I prefer it this way because it produces better debug logs if debugging is enabled. Not calling enter would only log that the hook was skipped, but not why. To me this worth the cost of checking for the state file twice even if it triggers some reader's desire to optimise the "useless" call away.

libexec/rc/rc.d/wireguard
687

Depending on your editor whitespace settings it arranges the code into a 2D table which helped me reason about it and saved vertical screen space for split editing. Is there a canonical style(9) like guide for shell instead of C to follow on how to reformat this?

rpokala added inline comments.
libexec/rc/rc.d/wireguard
687

I'm not sure if it's in style(9), but the mere fact that tab can be interpreted in different ways, suggests that space should be used.

Resigning from this; I tried to provide feedback over IRC, but that was seemingly not well-received (and questions unanswered) and I'm not interested in reviewing this as-is. I'd much prefer splitting it into two scripts, one with, e.g., verbs, that manages wireguard interfaces and then the rc script that simply drives that in an obvious way. The last objection I heard was that there's too much state to pass around, but it's not at all clear why unless this is trying to mix way too much rc.conf configuration in with wg config.

libexec/rc/rc.d/wireguard
687

Spaces can't be used because shell heredocs can only strip tabs not spaces from the beginning of lines.