-
Notifications
You must be signed in to change notification settings - Fork 38
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
base: main
Are you sure you want to change the base?
feat: improve commands_extension #1937
Conversation
This PR looks really good to me except that one thing I commented about :) |
There was a problem hiding this 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
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. |
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. 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, |
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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
There was a problem hiding this comment.
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
There was a problem hiding this 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 😊
b0fab79
to
33ee140
Compare
33ee140
to
b323617
Compare
@@ -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 { |
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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).
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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:
- Dartdoc comments that clearly explains this context wherever
stdout
is mentioned. - 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.
There was a problem hiding this 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
5f2a751
to
5b2f809
Compare
@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 :) |
ac2f5c8
to
d98052b
Compare
- 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]>
Signed-off-by: The one with the braid <[email protected]>
d98052b
to
06ba0c4
Compare
Tests should now be fixed, @coder-with-a-bushido ! |
@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 = {}; |
There was a problem hiding this comment.
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?
While implementing some custom commands, I stumbled across some issues with the current command runner :
/join
,/dm
etc. require to be entered with aRoom
present as parameterI have the following proposals to address these issues in the SDK's command execution :
StringBuffer
as stdout-like output buffer for commandstypedef
for the command function signatureClient.addCommand
signature now takes an optionalStringBuffer
as second parameter