08-02-2022 03:29 PM
@cbutcher wrote:
Imagine instead that A.lvclass is a testing dummy class, and you can more easily imagine why you'd not need it in all users of A-interface.lvlib
This. If all my messages are interfaces, I can swap out any nested actor with a stub, usually one that redirects the message to my unit test VI.
Even if I have no plans to ever have another working actor inherit that set of messages, I would still use an interface just for this purpose.
08-02-2022 03:35 PM
@justACS wrote:
@horuable wrote:
Ok, but if A (root) is already sending some messages to B (nested) without using interfaces, just normal messages from B's "Messages for this Actor" folder doesn't that alone make it so A cannot be used without B?
It does. The interfaces to A still do not belong in B. You may (for example) want to send some of those messages from another actor (A's caller, perhaps, or a new nested actor). That new actor should not have to depend on B's library in any way.
I'm not sure I understand this part. Couldn't you just create a new interface for that?
08-02-2022 03:44 PM
@justACS wrote:Personally, I don't even want A coupled to B. I put an interface on B as well, and have A call the interface. I have almost no coupling between my actors. Even if all it does is make it easy to isolate actors for unit testing, I count it as worth my tim
Not sure I agree with that (with the caveat that I'm not sure I understand it enough to be able to disagree with it).
Not coupling to your callers - absolutely. If write a low-level driver for some instrument, I absolutely want to be able to use it in other projects and have it be called by any actor. In that case, keeping the interface for what it sends to the caller as part of the driver makes sense to me. If I want to distribute it, I just distribute that one library. Any caller, just inherits from that interface and done.
Not coupling to the things you are calling, not entirely sure I understand the use case for that. If you are doing plugins and HALS, sure, but otherwise? Generally I know what I'm calling. And you are really only talking about middle layers here? right? because the top layer is probably application specific and not reusable anyway. So we're really only talking about reusing actors somewhere in the middle layer. How often are you actually reusing those? Wouldn't they have a lot of application specific logic in them anyway? Is there really that much to be gained by reusing them?
And also as far as HALS, do you really need a seperate actor for each instrument? or could you just have one actor that wraps the interface to your HAL using composition? Wouldn't that be easier? Then the only interface any other actors have to worry about is that one actor interface?
Also as to the unit testing, maybe that level of decoupling makes it easier, but is it required?
08-02-2022 03:45 PM
Not trying to be argumentative. Just having a really hard time following this conversation and I'm sure I'm I must be missing some important point that you and AQ are trying to make.
08-02-2022 04:16 PM
If the content is different then they aren’t sending the same data. I didn’t say data type.
08-02-2022 04:53 PM
> but shouldn't that imply that the interface should be a part of A (as
> in inside the same library), since A is always going to depend on it (the
> interface) anyway?
Whether discussing actors or not, I would suggest that it is a code smell if the same library owns both an interface and a class implementing that interface. I’m having trouble coming up with any case where packaging those together isn’t wrong other than the case of packaging for distribution in a packed project library, and I might question that one depending on circumstances.
08-03-2022 03:46 AM
@Taggart wrote:
@justACS wrote:Personally, I don't even want A coupled to B. I put an interface on B as well, and have A call the interface. I have almost no coupling between my actors. Even if all it does is make it easy to isolate actors for unit testing, I count it as worth my timNot coupling to the things you are calling, not entirely sure I understand the use case for that. If you are doing plugins and HALS, sure, but otherwise? Generally I know what I'm calling. And you are really only talking about middle layers here? right? because the top layer is probably application specific and not reusable anyway. So we're really only talking about reusing actors somewhere in the middle layer. How often are you actually reusing those? Wouldn't they have a lot of application specific logic in them anyway? Is there really that much to be gained by reusing them?
...
Also as to the unit testing, maybe that level of decoupling makes it easier, but is it required?
You can have the same reason here, like ACS wrote. If I can replace the called actor at will, I can trivially replace functions I don't like ("Spin Motor At Dangerous Speed.vi") with nicer versions (same name, but just log the command to a text file, or update a global variable, or something).
Then I can run my application whenever I want, and don't have to connect to hardware etc.
I can do that without an interface, via shared inheritance, but it's basically the same argument - if I need a parent class, I might as well have the interface (in 2020+), and if I plan to instead have my test class inherit from the real class, I have a whole range of potential problems (really hard breaking of LSP, plus probably I didn't use template method for a real class if that was the only goal, so no injection points for test code, etc...)).
@Taggart wrote:And also as far as HALS, do you really need a seperate actor for each instrument? or could you just have one actor that wraps the interface to your HAL using composition? Wouldn't that be easier? Then the only interface any other actors have to worry about is that one actor interface?
What's the purpose of the Actor here? I think you're saying this from the point of view of already having non-Actor drivers (class, library, bunch of VIs in a folder somewhere) to operate each piece of hardware (you mention HAL, but I'm not sure if you're talking about multiple implementations of a single tool, or multiple different types?).
Can you elaborate what you're (mentally, or really) doing?
After a few mental shufflings, I'm sort of imagining an Actor being used to make hardware operations asynchronous from the point of view of a calling Actor, so in the simplest (?) case, some ControllerActor.lvclass calling a WrapperActor.lvclass, which uses a MyAwesomeHardware.lvclass (not an Actor) to interact with some specific hardware.
You can replace with a HAL if you want more complexity, but I don't think it matters in the case you might be imagining, with a HAL for a single specific tool/behaviour, e.g. "Temperature Measuring At a Single Point Device"? - in that case maybe just a way to init the Actor with the appropriate concrete Temperature class, and then it's the same anyway.
You don't want ControllerActor to directly interact with Temperature, because it might take time, and it's supposed to be interacting with other messages etc, so you create WrapperActor.
In this mental model, it seems like WrapperActor has to be a "Temperature Measuring At a Single Point Device Wrapper Actor", and not a generic (Do Some Measurement) wrapper, but it doesn't need to be specific to the concrete Temperature class.
The level of wrapper here seems to depend entirely on the HAL/Interface - looks to me like one different Wrapper Actor per Interface / HAL that you intend to use.
If you really wanted to cut that down, you could use a message handling type arrangement (Do X.vi, X is a string/enum, case structure) to have an Actor depend on multiple interfaces/HALs and handle all of their behaviours, but mentally debugging that sounds like a pita... plus you're losing the asynchronous behaviour you wanted unless you program that lower down (out-of-band communication methods like non-Actor-Msg queues, notifiers, etc), in which case you might as well drop the wrapper and put it back in the ControllerActor anyway...
08-03-2022 04:38 AM
My train of thought is just leaving the station... I am now understanding that this discussion is more about interface packaging rather than interface segregation.
Regarding the argument of packaging an interface along with a class (actor or otherwise), I agree that is not advisable. If an interface is independent of classes (as it should be) it should be packaged in its own library along with its actor messages. Otherwise, what would the rationale be for selecting one class over another to package said interface?
If the interface is associated with only one or more related classes, then why even use an interface - just directly use the base class of these related classes and build the actor messages for this base class. One can still create stubs sub-classed from the base class for testing.
08-03-2022 08:39 AM - edited 10-31-2022 10:21 AM
Sam wrote:
> Not coupling to the things you are calling, not entirely sure I understand the use case for that.
Dhakkan wrote:
> If the interface is associated with only one or more related classes, then why even use an interface
Testing. You can replace the concrete dependency with a mock dependency during testing to simulate a wide range of responses from the dependency. For me, this is a case-by-case call of how testable I need each segment to be: if I can test lower levels and trust them when testing higher levels (and they aren’t particularly stateful), then I don’t tend to segregate… I can refactor later if I made the wrong call. JustACS, on the other hand, segregates always because ne finds having the interface segregation is cheap compared to the expense of refactoring if ne made the wrong call. It’s a reasonable position. But both of us agree that testing is the key value proposition for interfacing even a single-inheritor class.
08-03-2022 08:52 AM
@AristosQueue(NI) wrote:
> but shouldn't that imply that the interface should be a part of A (as
> in inside the same library), since A is always going to depend on it (the
> interface) anyway?
Whether discussing actors or not, I would suggest that it is a code smell if the same library owns both an interface and a class implementing that interface. I’m having trouble coming up with any case where packaging those together isn’t wrong other than the case of packaging for distribution in a packed project library, and I might question that one depending on circumstanc
I think you misunderstood what I was saying. If A is the caller and depends on the interface (which some other nested actor implements in order to send it messages) then it would make sense to package them (A and the interface) together. A is the consumer, not the implementation.