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

feat: improve commands_extension #1937

Open
wants to merge 4 commits into
base: main
Choose a base branch
from

Conversation

TheOneWithTheBraid
Copy link
Contributor

While implementing some custom commands, I stumbled across some issues with the current command runner :

  1. Currently, even Client-level commands such as /join, /dm etc. require to be entered with a Room present as parameter
  2. Currently, commands have no way to provide output, such as a success message, some reference to e.g. a newly created room or any other information since the return parameter is currently usually an eventId used by the clients to confirm sending a message was successful.
  3. Currently, [matrix] clients have no way to catch exceptions specific to the command execution, such as a missing or invalid parameter etc.

I have the following proposals to address these issues in the SDK's command execution :

  • unify behavior of all message sending related command
  • add a StringBuffer as stdout-like output buffer for commands
  • create a typedef for the command function signature
  • create a common exception type for command execution
  • enable commands to run on Client-level rather than Room-level
  • BREAKING: Client.addCommand signature now takes an optional StringBuffer as second parameter

@TheOneWithTheBraid TheOneWithTheBraid requested a review from a team as a code owner October 12, 2024 15:46
@coder-with-a-bushido
Copy link
Contributor

This PR looks really good to me except that one thing I commented about :)
Thank you so much!

Copy link
Contributor

@krille-chan krille-chan left a comment

Choose a reason for hiding this comment

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

May I ask where the issue is for this refactoring? Such a big change should go through a refinement first IMO

@TheOneWithTheBraid
Copy link
Contributor Author

May I ask where the issue is for this refactoring? Such a big change should go through a refinement first IMO

I'm not bound to any work processes when I'm contributing in my free time. The PR has half a page of documentation why and how the change was made. If you have question or would like to discuss the implementation, feel free to leave a constructive review comment. Since the API was not changed in a way relevant for client developers, I do not see the need of outlining a dedicated change proposal in advance.

@krille-chan
Copy link
Contributor

krille-chan commented Oct 17, 2024

May I ask where the issue is for this refactoring? Such a big change should go through a refinement first IMO

I'm not bound to any work processes when I'm contributing in my free time. The PR has half a page of documentation why and how the change was made. If you have question or would like to discuss the implementation, feel free to leave a constructive review comment. Since the API was not changed in a way relevant for client developers, I do not see the need of outlining a dedicated change proposal in advance.

But !WE! are bound to work processes! It would just have been nice to communicate such a change in advance instead of just expecting that we would spend our time in reviewing hundreds of code lines without knowing if it would have any benefit.

@TheOneWithTheBraid
Copy link
Contributor Author

Yeah, I see the problem. It's quite hard to have work processes on an open source open contribution project. Maybe some outline on starting from what amount of change (e.g. everything which is a new feature, changes API etc.) one should write an issue for discussion ? Might also help other external contributors ?

Currently the CONTRIBUTING.md only sais to open a merge request and discuss over there, that's why I so far did that for personal contributions, maybe you can update this (btw. it still sais GitLab too ^^).

@krille-chan
Copy link
Contributor

Yeah, I see the problem. It's quite hard to have work processes on an open source open contribution project. Maybe some outline on starting from what amount of change (e.g. everything which is a new feature, changes API etc.) one should write an issue for discussion ? Might also help other external contributors ?

Currently the CONTRIBUTING.md only sais to open a merge request and discuss over there, that's why I so far did that for personal contributions, maybe you can update this (btw. it still sais GitLab too ^^).

okay maybe I should rephrase my comments so far. Of course it is okay that everyone opens Pull Requests and we will always do our best to review them. The problem is only that we cannot guarantuee that the changes get merged. Especially for changes which change the API. Maybe we are not satisfied with how it looks like, the approach to fix it and so on.

The interest conflict here is mostly because external contributors have the desire to fix one specific thing, but usually don't have to deal with the code base after it, while we have to deal with the code base as it is our daily work. So we have a strong desire to keep the code base in a level of quality, which fits our needs.

For small changes this shouldn't be a big deal. For big changes, which apply to hundreds of code lines, it could mean, that we either request to change a lot of things so that it matches our interests as well, or we completely decline it at all. Then the whole work on the Pull Request would be nearly gone with the potential to leave the author back with some frustration.
Therefore we suggest to propose bigger changes in an issue first. Opening an issue, explaining the need and the idea how to solve it. Then we could comment if we think this is a good idea or give feedback before the implementation starts. This could drastically reduce the dangers that the Pull Request in the end gets merged.

In the end one could argue that it is not our problem but leaving pull request authors back with frustration might lead to shit talking about us. Especially from you I received a lot of FluffyChat pull requests where I felt a lot of pressure to merge them and even frustration when it took longer. This is something which could have been avoided.

I admit that my previous comment does not really reflected this and we should update our contribution guideline.

String msgtype = MessageTypes.Text,
String? threadRootEventId,
String? threadLastEventId,
StringBuffer? commandStdout,
Copy link
Contributor

Choose a reason for hiding this comment

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

Can we add a little bit dartdoc comments to make it more intuitive how to use the StringBuffer? Or at least a link to the documentation. For example I see this type for the first time and would have to look it up before using it.

Copy link
Contributor

Choose a reason for hiding this comment

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

We might want to have more information from the output as just a String, as errors could also be interesting, while we also might get errors and values at the same time. Could be probably solved by introducing a CommandResult type which contains the success data, (Matrix-)errors as well as the output of the command

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Hmmmm, in my implementation I simply used the regular error catching around the Client.parseAndRunCommand() and Room.send* methods for error handling. Since there are decent, unified Error types available I don't see an advantage in a dedicated error collection mechanism ; but if you have any proposal on how to implement some improved error handling, I'm curious for suggestions !

The advantage of a StringBuffer is that it basically allows to perform random access (similar to a File.open call) and object-oriented append syntax. I simply considered this as way more convenient for this use case - since a String could easily accidentally lose it's reference and doesn't support such a beautiful syntax to access its onwardly growing contents during the asynchronous writing into it.

Copy link
Contributor

Choose a reason for hiding this comment

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

@TheOneWithTheBraid Can we please have dartdoc comment for the commandStdout that says it's used for the command output in case a command was executed

Copy link
Contributor

Choose a reason for hiding this comment

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

@TheOneWithTheBraid ping

@TheOneWithTheBraid Can we please have dartdoc comment for the commandStdout that says it's used for the command output in case a command was executed

Copy link
Contributor

@krille-chan krille-chan left a comment

Choose a reason for hiding this comment

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

We looked at it on the refinement meeting together and we agree with the technical approach and the API changes. The usual review will follow. Thank you very much for the contribution 😊

@@ -91,8 +98,12 @@ extension CommandsClientExtension on Client {

/// Register all default commands
void registerDefaultCommands() {
addCommand('send', (CommandArgs args) async {
return await args.room.sendTextEvent(
addCommand('send', (args, stdout) async {
Copy link
Contributor

Choose a reason for hiding this comment

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

we could add the eventId in the stdout here?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I'd object this since sending an Event is the default "action" on the Room.send. The event ID always used to be returned as result of the method, not as command output. I'd rather see the standard output StrinBuffer as a complimentary concept for additional information which is exactly not the default behavior (aka sending an event).

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Consider the following flow :

As a user, I enter something into a message text field of my matrix client. I send the contents via the Room.send* methods.

The matrix client underneath now does not know at first if I just sent a regular message or if a command was executed. As the matrix client, I will know about this by checking the stdout. If there is data in the stdout, I consider the result as command result and show e.g. a dialog or a snack bar with a details button where I can see the stdout.

If the stdout is empty and I simply got back an event id as return value, as a matrix client, I simply know the user just sent a message.

If we'd by default add the event id to the stdout, the entire flow of determination whether a command with special result was executed or just a simple event was sent stops working.

Copy link
Contributor

Choose a reason for hiding this comment

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

I'm just trying to understand the use case for stdout once again, pls correct me if I'm wrong.

I use Room.send with a text while I'm unsure if it's actually a command or just a text that needs to be sent. So, what I do is I create a stdout object that will return me something (can be any meaningful data) when it's a command that does something other than sending an event.

If that's the case, as the user of the SDK I want 2 things to be done:

  1. Dartdoc comments that clearly explains this context wherever stdout is mentioned.
  2. An example usage of this scenario in both README and dartdoc comment above Room.send.

If I didn't understand it correctly, please tell me.

Copy link
Contributor

@coder-with-a-bushido coder-with-a-bushido left a comment

Choose a reason for hiding this comment

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

@TheOneWithTheBraid Sorry for the super late review 😖

Code looks really good to me except for those nitpick comments. Now that you've used the stdout in some commands, I somehow want to use it all commands possible so it can be used

@TheOneWithTheBraid TheOneWithTheBraid force-pushed the braid/command-runner branch 2 times, most recently from 5f2a751 to 5b2f809 Compare December 13, 2024 13:12
@TheOneWithTheBraid
Copy link
Contributor Author

@coder-with-a-bushido Thanks for the feedback ! I (hopefully) fixed the test and added two comments above, I'd appreciate some feedback. Have a good weekend !

@coder-with-a-bushido
Copy link
Contributor

@coder-with-a-bushido Thanks for the feedback ! I (hopefully) fixed the test and added two comments above, I'd appreciate some feedback. Have a good weekend !

Hey, thank you. But looks like 1 test is still broken and I think 2 more comments should be resolved before we can merge this :)

- unify behavior of all message sending related command
- add a StringBuffer as stdout-like output buffer for commands
- create a typedef for the command function signature
- create a common exception type for command execution
- enable commands to run on Client-level rather than Room-level
- BREAKING: Client.addCommand signature now takes an optional
  StringBuffer as second parameter

Signed-off-by: The one with the braid <[email protected]>
Signed-off-by: The one with the braid <[email protected]>
Signed-off-by: The one with the braid <[email protected]>
@TheOneWithTheBraid
Copy link
Contributor Author

Tests should now be fixed, @coder-with-a-bushido !

@TheOneWithTheBraid
Copy link
Contributor Author

@coder-with-a-bushido Can you elaborate on what's wrong with the coverage test ?

@coder-with-a-bushido
Copy link
Contributor

@coder-with-a-bushido Can you elaborate on what's wrong with the coverage test ?

I don't understand it either. The test is breaking in two totally different files, let me try it locally first. Meanwhile, can you pls address the 2 other comments left?

@@ -98,7 +98,7 @@ class Client extends MatrixApi {
DateTime? _accessTokenExpiresAt;

// For CommandsClientExtension
final Map<String, FutureOr<String?> Function(CommandArgs)> commands = {};
final Map<String, CommandExecutionCallback> commands = {};
Copy link
Contributor

Choose a reason for hiding this comment

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

Can you also pls add "BREAKING" in the "feat: improve command_extension" commit message?

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.

3 participants