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

Mark as resolved/unresolved prep 1 #1242

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

Conversation

chrisbobbe
Copy link
Collaborator

My local branch for #744 has gotten pretty long and I think it'll benefit from being split into multiple PRs. 🙂 Here's one.

@chrisbobbe chrisbobbe added the integration review Added by maintainers when PR may be ready for integration label Dec 30, 2024
@chrisbobbe chrisbobbe requested a review from gnprice December 30, 2024 21:49
Copy link
Member

@gnprice gnprice left a comment

Choose a reason for hiding this comment

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

Thanks! Good idea.

Comments below from an initial review.

Comment on lines 166 to 169
String stripResolvePrefixIfPresent(String topic) =>
topic.startsWith(kResolvedTopicPrefix)
? topic.substring(kResolvedTopicPrefix.length)
: topic;
Copy link
Member

Choose a reason for hiding this comment

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

How does this end up getting used later in your branch?

Its behavior differs from unresolve_name in the shared code:
https://github.com/zulip/zulip/blob/1fac99733/web/shared/src/resolved_topic.ts#L22-L29

which I'm guessing (haven't checked) is what gets applied when the user unresolves a topic in web or the legacy mobile app.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Updating to match web's behavior.

Comment on lines +69 to +73
/// When built from server data, the action sheet ignores changes in that data;
/// we intentionally don't live-update the buttons on events.
/// If a button's label, action, or position changes suddenly,
/// it can be confusing and make the on-tap behavior unexpected.
/// Better to let the user decide to tap
Copy link
Member

Choose a reason for hiding this comment

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

Hmm, yeah.

Looking for PerAccountStoreWidget.of calls in this file, I think there's one place we don't currently follow this: the hasSelfVote in ReactionButtons.

That one shouldn't cause this problem in practice, though, because it changes by the user's own action. (Or by the message getting deleted, but then it doesn't matter whether the user tries to add a reaction or remove one.)

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

In a quick manual test, I don't even see the hasSelfVote live-updating in the message action sheet as I add or remove one of the popular emojis from the web app.

hasSelfVote uses PerAccountStoreWidget.of to get the self-user's ID. My user ID doesn't change when I add/remove a reaction. And although we're still signing up to refresh whenever the PerAccountStore notifies listeners, it doesn't notify listeners on ReactionEvents. (Any MessageListViews do notify their listeners.)

Copy link
Member

Choose a reason for hiding this comment

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

In a quick manual test, I don't even see the hasSelfVote live-updating in the message action sheet as I add or remove one of the popular emojis from the web app.

Right, but then it's still your own action — you know that you just made that change (even though via a different client). So there isn't a risk of confusion.

Comment on lines +73 to +76
/// Better to let the user decide to tap
/// based on information that's comfortably in their working memory,
/// even if we sometimes have to explain (where we handle the tap)
/// that that information has changed and they need to decide again.
Copy link
Member

Choose a reason for hiding this comment

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

A key point here is that these races are always possible anyway — the change may have happened on the server but not reached us yet.

@@ -1,5 +1,6 @@
import 'package:json_annotation/json_annotation.dart';

import '../route/messages.dart';
Copy link
Member

Choose a reason for hiding this comment

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

Hmm, I don't like having lib/api/model/ import from lib/api/route/. If it's needed for the model, that feels like a sign it's part of the model and not part of how particular routes interact with the model.

I think these can just stay in this file.

Copy link
Member

@gnprice gnprice Dec 30, 2024

Choose a reason for hiding this comment

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

(When this file gets to feel unwieldy, I think the thing to do is to split pieces out by subject area: like a lib/api/model/message.dart, akin to the narrow.dart and submessage.dart and so on that we already have.)

Comment on lines -769 to -771
/// As in [UpdateMessageEvent.propagateMode].
@JsonEnum(fieldRename: FieldRename.snake)
enum PropagateMode {
Copy link
Member

Choose a reason for hiding this comment

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

And this can move from lib/api/model/events.dart to lib/api/model/model.dart, since it's no longer specific to an event.

@chrisbobbe chrisbobbe force-pushed the pr-resolve-unresolve-prep-1 branch from fe11d69 to 68d7cd1 Compare January 16, 2025 01:04
@chrisbobbe
Copy link
Collaborator Author

Thanks for the review! Revision pushed.

Comment on lines 859 to 896
if (topic.apiName.startsWith(_resolvedTopicPrefix)
&& topic.apiName.substring(_resolvedTopicPrefix.length) == prevTopic.apiName) {
return true;
}

if (prevTopic.apiName.startsWith(_resolvedTopicPrefix)
&& prevTopic.apiName.substring(_resolvedTopicPrefix.length) == topic.apiName) {
if (topic.isSameAs(prevTopic, ignoringResolvePrefix: false)) { // TODO(log)
// Not a topic move after all… Return `true`
// just because `false` would put a marker for `MessageEditState.moved`
// on a message in the message list, which would be conspicuously wrong.
return true;
}

return false;
return topic.isSameAs(prevTopic, ignoringResolvePrefix: true);
Copy link
Member

Choose a reason for hiding this comment

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

Because this operation is something we do when decoding one of the high-volume types of data we get from the server (namely messages), it's an area where I'd like to stay somewhat paranoid about performance. (Similar to #1156 (comment) and other comments I've made in some PRs about parsing message content.)

Concretely, that means:

  • Let's have this function call canonicalize only once on each of the topics, not twice.
    • (Even then the loop in _readFromMessage will end up calling it twice on each of the topics in the middle of the history, once by passing it as prevTopic here and once as topic. Ideally we'd fix that too, but that might be more annoying and I'm content to let it be. That issue compounds with this one, though — it means that in this revision it calls canonicalize four times on each intermediate topic.)
  • It's a bit regrettable this needs a regexp instead of the startsWith/substring check. We do want to get the semantics right, though. Is that definitely what web does?

Copy link
Collaborator Author

@chrisbobbe chrisbobbe Jan 16, 2025

Choose a reason for hiding this comment

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

It's a bit regrettable this needs a regexp instead of the startsWith/substring check. We do want to get the semantics right, though. Is that definitely what web does?

Oh! Actually this isn't what web does. The code before matches what web does. Compare (zulip-flutter before vs. current web):

  static bool topicMoveWasResolveOrUnresolve(TopicName topic, TopicName prevTopic) {
    if (topic.apiName.startsWith(TopicName.resolvedTopicPrefix)
        && topic.apiName.substring(TopicName.resolvedTopicPrefix.length) == prevTopic.apiName) {
      return true;
    }

    if (prevTopic.apiName.startsWith(TopicName.resolvedTopicPrefix)
        && prevTopic.apiName.substring(TopicName.resolvedTopicPrefix.length) == topic.apiName) {
      return true;
    }

    return false;
  }
                // We know it has a topic edit. Now we need to determine if
                // it was a true move or a resolve/unresolve.
                if (
                    resolved_topic.is_resolved(edit_history_event.topic) &&
                    edit_history_event.topic.slice(2) === edit_history_event.prev_topic
                ) {
                    // Resolved.
                    resolve_toggled = true;
                    continue;
                }
                if (
                    resolved_topic.is_resolved(edit_history_event.prev_topic) &&
                    edit_history_event.prev_topic.slice(2) === edit_history_event.topic
                ) {
                    // Unresolved.
                    resolve_toggled = true;
                    continue;
                }
                // Otherwise, it is a real topic rename/move.
                moved = true;

For this "MOVED"-marker logic, there are pros and cons of following web:

  • Pro: performance, as you mentioned
  • Pro (I guess): You get a "MOVED" marker if the topic changes from like "✔ ✔ ✔ some topic" to "✔ ✔ some topic"
  • Con: You still get a "MOVED" marker on an actual resolve/unresolve, if the resolved-topic prefix is weird: "✔ ✔ some topic" to "some topic"
  • Con: Case-folding is not done

Copy link
Collaborator Author

@chrisbobbe chrisbobbe Jan 16, 2025

Choose a reason for hiding this comment

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

It looks like web uses the regex in just one place: to make an unresolve-topic API request. That seems sensible; as you mentioned in the office, the output of "unresolve topic" shouldn't be a resolved topic or look like one.

Copy link
Member

Choose a reason for hiding this comment

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

Cool. Yeah, let's stick with the simpler existing behavior which matches web, then.

The job of this "was resolve or unresolve" function is really just to paper over the fact that our "resolve topic" feature is implemented under the hood by renaming the topic. So it's fine that if someone edits the topic name only by fixing the letter case, then that will be displayed with a "MOVED" marker — the user who made that change indeed did so through a "move this topic" UI, just like for other times we show a "MOVED" marker.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Oh I see, yeah, that makes sense.

Comment on lines 890 to 898
if (topic.isSameAs(prevTopic, ignoringResolvePrefix: false)) { // TODO(log)
// Not a topic move after all… Return `true`
// just because `false` would put a marker for `MessageEditState.moved`
// on a message in the message list, which would be conspicuously wrong.
return true;
Copy link
Member

Choose a reason for hiding this comment

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

I guess further on the point of performance paranoia: is this a situation that the server is actually expected to produce?

I think not ­— https://zulip.com/api/get-messages at prev_topic says:

Only present if message's topic was edited.

So we don't need to go out of our way trying to detect it.

///
/// If [ignoringResolvePrefix] is true,
/// the [unresolve]d forms of [this] and [other] are compared instead.
bool isSameAs(TopicName other, {required bool ignoringResolvePrefix}) {
Copy link
Member

Choose a reason for hiding this comment

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

This feels to me more like two different methods than one method with a boolean flag.

And then I think the "ignoring" case is basically doing much the same job as its one caller topicMoveWasResolveOrUnresolve, so can be inlined into there.

Comment on lines +69 to +73
/// When built from server data, the action sheet ignores changes in that data;
/// we intentionally don't live-update the buttons on events.
/// If a button's label, action, or position changes suddenly,
/// it can be confusing and make the on-tap behavior unexpected.
/// Better to let the user decide to tap
Copy link
Member

Choose a reason for hiding this comment

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

In a quick manual test, I don't even see the hasSelfVote live-updating in the message action sheet as I add or remove one of the popular emojis from the web app.

Right, but then it's still your own action — you know that you just made that change (even though via a different client). So there isn't a risk of confusion.

Copy link
Member

@gnprice gnprice left a comment

Choose a reason for hiding this comment

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

Thanks for the revision! Generally this all looks good. A few comments above, and a couple more below.

changeLater,
changeAll;

String toJson() =>_$PropagateModeEnumMap[this]!;
Copy link
Member

Choose a reason for hiding this comment

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

nit:

Suggested change
String toJson() =>_$PropagateModeEnumMap[this]!;
String toJson() => _$PropagateModeEnumMap[this]!;

Future<UpdateMessageResult> updateMessage(
ApiConnection connection, {
required int messageId,
String? topic,
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
String? topic,
TopicName? topic,

🙂

@chrisbobbe chrisbobbe force-pushed the pr-resolve-unresolve-prep-1 branch from 68d7cd1 to ee4f762 Compare January 16, 2025 22:27
@chrisbobbe
Copy link
Collaborator Author

Thanks for the review! Revision pushed, and please see #1242 (comment).

@gnprice
Copy link
Member

gnprice commented Jan 16, 2025

Thanks! All looks good except that open subthread above.

Instead of passing its fields separately. (We'd like to start using
lastUnreadId from the data too.)
This will make it available in a long-press handler in the app bar
for a topic narrow. That long-press handler brings up the topic
action sheet, which will need a message ID to pass in the API
request for resolve/unresolve topic. (Not sure the API request
should *need* to require a message ID, but it does; shrug.)
And move the PropagateMode definition from events.dart to
model.dart, since it's no longer specific to an event.
@chrisbobbe chrisbobbe force-pushed the pr-resolve-unresolve-prep-1 branch from ee4f762 to 3ef535f Compare January 17, 2025 23:26
@chrisbobbe
Copy link
Collaborator Author

Thanks for the review! Revision pushed, atop #1290 for the sake of CI.

Copy link
Member

@gnprice gnprice left a comment

Choose a reason for hiding this comment

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

Thanks for the revision! One comment now on the details of unresolve.

Comment on lines +139 to +140
doTest(eg.t('some topic'), eg.t('some topic'), expectIdentity: true);
doTest(eg.t('Some Topic'), eg.t('Some Topic'), expectIdentity: true);
Copy link
Member

Choose a reason for hiding this comment

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

This ends up checking that the 'some topic' on the left is identical to the 'some topic' on the right. From the docs at [identical] I think that isn't guaranteed to be true.

Conversely, it seems like the point of this identity check is to test the return this; case in the implementation. But the test actually still passes if you remove that:

    if (match == null) {
      return TopicName(_value.substring(0));
    } else {
      return TopicName(_value.substring(match.end));
    }

or even replace the whole function with a String.replaceFirst call:

    return TopicName(_value.replaceAll(resolvedTopicPrefixRegexp, ''));

Let's simplify to the latter, and leave out the identicalTo check. We'll rely on the Dart stdlib to have the optimization where it doesn't unnecessarily copy the string in replaceFirst when there were no matches to replace.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
integration review Added by maintainers when PR may be ready for integration
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants