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

RSDK-9440 Report machine state through GetMachineStatus #4616

Merged
merged 32 commits into from
Jan 9, 2025

Conversation

benjirewis
Copy link
Member

@benjirewis benjirewis commented Dec 10, 2024

RSDK-9440

Changes:

  • Adds a new enumerated State field to robot.MachineStatus both server and client side
  • Starts machines with a "minimal" config to start web service earlier before starting with full config
  • Reports StateInitializing in robot.MachineStatus before reconfigure with full config occurs
  • Reports StateRunning in robot.MachineStatus after reconfigure with full config occurs
  • Exposes a SetInitializing method on robot.LocalRobot for the above two points to work

Testing:

  • Modifies server and client side MachineStatus tests to make assertions on State
  • Basic integration test
  • Checks for state running before returning from client.New when in testing
  • Adds more injected machine status functions to client and client session tests

@viambot viambot added the safe to test This pull request is marked safe to test from a trusted zone label Dec 10, 2024
@benjirewis benjirewis changed the title RSDK-9440 RSDK-9440 Report machine state through GetMachineStatus Dec 10, 2024
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Dec 10, 2024
Copy link
Member Author

@benjirewis benjirewis left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Still a WIP wrt testing; will leave in draft. These are my ideas so far, though.

web/server/entrypoint.go Outdated Show resolved Hide resolved
web/server/entrypoint.go Outdated Show resolved Hide resolved
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Dec 11, 2024
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Dec 11, 2024
@benjirewis
Copy link
Member Author

I broke a lot of tests that I'm presuming are expecting resources to be available as soon as the web service is available. Thinking about it.

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Dec 13, 2024
cli/client_test.go Outdated Show resolved Hide resolved
robot/client/client.go Outdated Show resolved Hide resolved
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Dec 13, 2024
minimalProcessedConfig.Modules = nil
minimalProcessedConfig.Processes = nil

myRobot, err := robotimpl.New(ctx, minimalProcessedConfig, s.logger, robotOptions...)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this the best way of achieving what we want or the most expedient?

I'm fine with this as-is. And I'm kind of fine never coming back to think about this. But the whole "robot owns the web server" feels backwards.

There would be less states to consider if we could start a web service and register robots with it. And there'd be a small API that describes:

  • What state the robot is in (startup or running) and
  • which APIs are available, e.g:
    • just "GetMachineStatus" and maybe "ResourceNames"
    • but none of "SetPower"/other resource specific APIs

But in this PR we have 90-100 lines between this comment/robot.New and when the web service is started. That's a lot of lines to accidentally break our contract and add some blocking code.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this the best way of achieving what we want or the most expedient?

Perhaps not, and I understand your argument there. I've introduced a slightly different/simpler mechanic for controlling the "initializing" value, so that might address some of your concerns here. I didn't go so far as starting a web service and registering robots with it (if I'm understanding what you're saying.)

web/server/entrypoint.go Outdated Show resolved Hide resolved
@@ -479,7 +502,8 @@ func (s *robotServer) serveWeb(ctx context.Context, cfg *config.Config) (err err
}()
defer cancel()

options, err := s.createWebOptions(processedConfig)
// Create initial web options with `minimalProcessedConfig`.
options, err := s.createWebOptions(minimalProcessedConfig)
Copy link
Member

@dgottlieb dgottlieb Dec 16, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I can't comment off-diff. The goroutine spun off above will check for diff.NetworkEqual and if not, run myRobot.StartWeb (newline 490).

Just below this we call web.RunWeb. I'm not sure what the significance is between having different methods, StartWeb and RunWeb, but assuming that's not interesting: is it possible for those two things to race? Are we guaranteed to end up with the right set of weboptions?

To clarify, this is a question about existing behavior. I don't think this patch changed anything here.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The goroutine spun off above will check for diff.NetworkEqual and if not, run myRobot.StartWeb (newline 490).

Correct; I believe there's a call to StopWeb before that happens, too. And then a Reconfigure after that StopWeb. All about handling network changes in the config.

Just below this we call web.RunWeb. I'm not sure what the significance is between having different methods, StartWeb and RunWeb, but assuming that's not interesting

It's interesting having those two methods. StartWeb starts up the web service on the robot. RunWeb does that, but also waits on <-ctx.Done(), so it's a blocking call and represents the "main" program that "runs" when you call go run web/cmd/server/main.go.

Is it possible for those two things to race? Are we guaranteed to end up with the right set of weboptions?

It depends what you mean by race. There is a lock on starting the web service, so I'm not sure we'd see a race manifest as an actual DATA RACE, but I think you are "right" in wondering about the "right set of weboptions." I'm not entirely sure, but I think RunWeb would run into an error if it tried to start the web service with an old set of options after the config watcher goroutine had started it already with a new set of options. So, my guess is we'd see an error from RunWeb and an inability to start the server in the event of the race you're describing.

@@ -498,6 +502,8 @@ func newWithResources(
}

successful = true
// Robot is "initializing" until first reconfigure after initial creation completes.
r.initializing.Store(true)
Copy link
Member

@dgottlieb dgottlieb Dec 16, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Technically there's a Reconfigure earlier on newcode 491 that sets this value to false. Not to mention the initializing value is initialized (ugh) to false. Two things:

  • I'm taking that it's important we exit this function with initializing set to true. But I would expect to see the setting up at the top near the constructor. Can we document that the placement here is intentional to avoid prior calls mucking with the state?
  • Is this function guaranteed to not start webserver and expose the GetMachineStatus API? If it can, it seems we might be setting initializing too late and may allow clients to observe an illegal transition of ready -> initializing.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also -- line 428 (old) 432 (new) refers to the mod manager web server. Do we need to consider/provide guidelines for how module SDKs (which are -- in theory -- different from "application SDKs") use and perhaps expose GetMachineStatus?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've modified the mechanics here slightly. You'll see there's a new robot.Option to start a robot in initializing mode. You can then use SetInitializing(false) to mark the robot as running. This means that only the code here in web/server/entrypoint.go is "special" with respect to initialization. All other calls to robotimpl.New will create robots that always return robot.StateRunning from MachineStatus.

I'm not sure that will address all your concerns here, and I'll think a bit harder about your module question before re-requesting review.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we need to consider/provide guidelines for how module SDKs (which are -- in theory -- different from "application SDKs") use and perhaps expose GetMachineStatus?

Slightly confused about your question here. The mod manager web server is started before any modules are added to the module manager, so it should always be the case that "module SDKs" (I'm reading that as Golang, Python, and C++ module libraries) should have the ability to connect back to the RDK through the mod manager web server before any module process has started. Were you suggesting that the module libraries should be using MachineStatus to check the status of the parent RDK, or that they should expose their own MachineStatus endpoint for some reason?

Copy link
Member

@dgottlieb dgottlieb Dec 20, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Slightly confused about your question here. The mod manager web server is started before any modules are added to the module manager, so it should always be the case that "module SDKs" (I'm reading that as Golang, Python, and C++ module libraries) should have the ability to connect back to the RDK through the mod manager web server before any module process has started.

Given that, it sounds like if* a module, as soon as it was possible, tried calling MachineStatus they'd get an initialized == false.

But it also sounds like in the current, pre-patch code, a module could actually call ResourceNames before a regular "network client" could? Because we wouldn't have started accepting connections yet? And the result of calling ResourceNames would be undefined as the robot hasn't necessarily done/completed its initial reconfigure yet? Just considering the "happy path" where the initial robot config is good.

And if that's true, modules already have to be "resilient" to talking with an "uninitialized" robot. And we would not expect to need module changes. Such as the testing changes to "wait by default" for a robot to be initialized.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Given that, it sounds like if* a module, as soon as it was possible, tried calling MachineStatus they'd get an initialized == false.

That sounds correct to me.

But it also sounds like in the current, pre-patch code, a module could actually call ResourceNames before a regular "network client" could? Because we wouldn't have started accepting connections yet?

That sounds correct to me. In particular, the "module web server" of the RDK will be open for connections while the regular web server will not be. So, a module could connect via the former before a regular "network client" could connect via the latter.

And the result of calling ResourceNames would be undefined as the robot hasn't necessarily done/completed its initial reconfigure yet?

That does not sound correct to me. If a module is able to call ResourceNames through the module server, it will see the current status of the resource graph in terms of available names. If some resources have already completed configuration, the module will see them through ResourceNames.

And if that's true, modules already have to be "resilient" to talking with an "uninitialized" robot. And we would not expect to need module changes.

Modules do expect a certain guarantee around modular dependencies: if a modular resource A depends on another resource B, it is expected that, barring any inability to create B, B will be available and usable through ResourceByName within the constructor for A. I actually broke that guarantee and caused TestComplexModule to fail here.

We need the web service, which is "weakly" (it's actually, annoyingly, a fourth type of hardcoded-weak dependency that is not registered via WeakDependencies in resource registration) dependent on all resources to Reconfigure before the modular base builds such that the web service is aware of the two motors that the modular base depends on.

I "fixed" the test by updating weak dependents more often and more simply in completeConfig (see my incoming comment.)

// been closed above. This ensures processes are shutdown before any files
// are deleted they are using.
//
// If initializing, machine will be starting with no modules, but may
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure I understand this. Does anything actually go wrong if we don't guard this "cleanup" logic? Or are we just suggesting that making these calls would be "wasteful" no-ops?

The existing comment/first paragraph refers to "cleanup unused packages", so if we never started up with any, where are they coming from? Existing files on the file system from a previous start?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@cheukt had mentioned it would be good to guard the lines below based on initialization.

Or are we just suggesting that making these calls would be "wasteful" no-ops?

That's my understanding, yep.

Existing files on the file system from a previous start?

Also my understanding, yep.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

if we don't guard this call, I suspect that we would cleanup modules from the full config that aren't in the initial minimal config, so we end up deleting modules that would be used. If offline, the robot would no longer be able to re-download the module and start up correctly.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think I understand. Because we're literally trying to start the server with an empty config, this code will delete all the things unless we communicate/check the special "this isn't a real reconfigure step" initializing flag.

Maybe this is worth clarifying -- but when we were talking about "this feature should appear as if the robot was first configured as if it had no components, followed by a reconfigure with components": I can't tell if we went with implementing this that way because it was the best way to solve the problem? Or if we went down that path because the above statement was interpreted as "should literally be implemented as" a bit?

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Dec 16, 2024
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 3, 2025
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 3, 2025

// Set value for `DoNotWaitForRunningEnvVar` to allow connecting to a
// still-initializing machine.
test.That(t, os.Setenv(client.DoNotWaitForRunningEnvVar, "true"), test.ShouldBeNil)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've always heard about how it's unsafe to change environment variables -- particularly in a multi-threaded program. I don't think I understand the claim well enough to know whether there's a problem here. Instead I'll ask:

  • does this need to be an environment variable because we need to control processes outside of the test program?
  • Or would a global variable do just as well?

Seeing that this env variable lookup happens alongside a test.IsTesting(), I'm assuming the latter.

I could also see value in letting users disable the behavior with a client dial option. But I don't personally need to see that in this PR.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great point; it's the latter, and I've switched to using a global atomic boolean.

if status.Code(err) == codes.Unimplemented {
break
}
return nil, multierr.Combine(err, rc.conn.Close())
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm going to recommend ignoring the error from rc.conn.Close() and just returning the err from rc.MachineStatus instead.

I've been undoing this pattern as I've come across it. Especially when correctness doesn't require Close to be called. See: the webrtc connection bugs where a connection succeeded, but we disconnected from the peer because SendDone[1] to the signaling server failed.

[1] I claim SendDone is analogous to Close.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Gotcha that sounds good to me; thanks for pointing it out.

@@ -129,13 +132,11 @@ func TestClientSessionOptions(t *testing.T) {
Disable: true,
})))
}
roboClient, err := client.New(ctx, addr, logger, opts...)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

client.New now calls MachineStatus in testing environments, which is not exempted from session creation.

I'm fine not touching this in the current PR. But are we currently aware of a reason why MachineStatus should not be exempt?

minimalProcessedConfig.Processes = nil

// Start robot in an initializing state with minimal config.
robotOptions = append(robotOptions, robotimpl.WithInitializing())
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What's the consequence of not having this line?

  • Do we go back to the old behavior of having to do an initial reconfigure before opening a server port?
  • Or do we still open a server port before configuring -- but we just lie to clients that the robot is in "MachineState.RUNNING"?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Or do we still open a server port before configuring -- but we just lie to clients that the robot is in "MachineState.RUNNING"?

This one is the consequence; the robot would falsely report itself as running.

// been closed above. This ensures processes are shutdown before any files
// are deleted they are using.
//
// If initializing, machine will be starting with no modules, but may
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think I understand. Because we're literally trying to start the server with an empty config, this code will delete all the things unless we communicate/check the special "this isn't a real reconfigure step" initializing flag.

Maybe this is worth clarifying -- but when we were talking about "this feature should appear as if the robot was first configured as if it had no components, followed by a reconfigure with components": I can't tell if we went with implementing this that way because it was the best way to solve the problem? Or if we went down that path because the above statement was interpreted as "should literally be implemented as" a bit?

// Once reconfigure with initial config is complete; set initializing to
// false. Robot is now fully running and can indicate this through the
// MachineStatus endpoint.
r.SetInitializing(false)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Because I see we had to expose a whole method for this on the LocalRobot API, I feel compelled to ask: would it be wrong to have localRobot.Reconfigure just set this value at the end?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

newWithResources calls Reconfigure to construct the robot, so if we set initializing to false at the end of Reconfigure, then initializing would be incorrectly false before the Reconfigure above completes.

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 3, 2025
@dgottlieb
Copy link
Member

@benjirewis and I talked offline about options regarding managing the initializing flag. Broadly, I'm fine with how the code is now, but @benjirewis was on board with taking a soft pass at some ideas for ensuring (in the future as developers continue to muck with the code) we never violate the invariant of having a configured robot, but also have initializing = true.

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 6, 2025
Copy link
Member Author

@benjirewis benjirewis left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Per @dgottlieb 's comment, I've moved away from WithInitializing and SetInitializing and opted to just put an Initial field on config.Config instead. The logic around is hopefully somewhat self-explanatory at this point, but basically I've just simplified the way the entrypoint code interacts with the localRobot.initializing atomic boolean.

@@ -129,13 +132,11 @@ func TestClientSessionOptions(t *testing.T) {
Disable: true,
})))
}
roboClient, err := client.New(ctx, addr, logger, opts...)
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

are we currently aware of a reason why MachineStatus should not be exempt?

As far as I can tell, we only rely on MachineStatus starting sessions for tests, and, while it's not currently, it should be exempt. I assume we'll handle this eventually as part of RSDK-9262.

// If initializing, machine will be starting with no modules, but may
// immediately reconfigure to start modules that have already been
// downloaded. Do not cleanup packages/module dirs in that case.
if !r.initializing.Load() {
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added some assertions to TestMachineState in web/server/entrypoint_test.go to ensure local packages get (or do not get) cleaned up. Manually tested that start-up of a robot does not remove existing cloud or module manager packages.

@@ -192,3 +193,54 @@ func isExpectedShutdownError(err error, testLogger logging.Logger) bool {
testLogger.Errorw("Unexpected shutdown error", "err", err)
return false
}

// Tests that machine state properly reports initializing or running.
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Modified the test below to test the transition.

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 7, 2025
@@ -101,6 +107,7 @@ type configData struct {
LogConfig []logging.LoggerPatternConfig `json:"log,omitempty"`
Revision string `json:"revision,omitempty"`
MaintenanceConfig *MaintenanceConfig `json:"maintenance,omitempty"`
PackagePath string `json:"package_path,omitempty"`
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We were missing some code to marshal and unmarshal the PackagePath field. I needed that functionality for TestMachineState, so I just added it.

out.AllowInsecureCreds = s.args.AllowInsecureCreds
out.UntrustedEnv = s.args.UntrustedEnv

// Use ~/.viam/packages for package path if one was not specified.
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added this if-check to not forcibly change the initial package path to be ~/.viam/packages and respect whatever was unmarshaled from disk if there was anything (non-"".)

@benjirewis benjirewis requested review from cheukt and dgottlieb January 7, 2025 22:12
Copy link
Member

@cheukt cheukt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM! a few small questions/nits

//
// Allow this behavior to be turned off in some tests that specifically want
// to examine the behavior of a machine in an initializing state through the
// use of an environment variable.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// use of an environment variable.
// use of a package variable.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good catch; applied something similar locally.

robot/client/client.go Show resolved Hide resolved
config/config.go Outdated
// Initial represents whether this is an "initial" config passed in by web
// server entrypoint code. If true, the robot will continue to report a state
// of initializing after applying this config. If false, the robot will
// report a state of reconfiguring after applying this config.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// report a state of reconfiguring after applying this config.
// report a state of running after applying this config.

?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

😮‍💨 good catch; applied locally.

@@ -19,6 +19,7 @@ type options struct {
// shutdownCallback provides a callback for the robot to be able to shut itself down.
shutdownCallback func()

// whether FTDC is enabled
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Did the linter complain? Because I really don't want to nitpick that the tense of enabled implies the member is describing what is currently running -- as opposed simply describing the users intent for when the robot is started!

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No idea why I added this (was not linter complaining); switched to whether or not to run FTDC. Hopefully that captures what you're getting at here, but let me know if not.

close(onWatchDone)
})
onWatchDone := make(chan struct{})
go func() {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If I'm following correctly, this changed from a ManagedGo to a raw goroutine. Should I be scrutinizing this?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Correct and no I wouldn't scrutinize too deeply. Sometimes I find that ManagedGo only slightly obfuscates what's going on, and I switched to a regular goroutine here because I felt it was more natural for me to write.

In reality, and if I'm reading correctly, ManagedGo is a wrapper around PanicCapturingGoWithCallback that allows specifying a callback (accomplished with a defer here) and just re-panics if a panic is encountered (no panic capturing.)

onWatchDone := make(chan struct{})
go func() {
defer func() {
close(onWatchDone)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If we're leaving this as a raw goroutine -- can go straight to defer close(onWatchDone).

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yep.

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 9, 2025
@benjirewis benjirewis merged commit 4775e98 into viamrobotics:main Jan 9, 2025
16 checks passed
@benjirewis benjirewis deleted the machine-state branch January 9, 2025 17:41
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
safe to test This pull request is marked safe to test from a trusted zone
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants