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

local echo (1/n): Add API support for local_id/local_message_id and queue_id #1454

Merged
merged 5 commits into from
Apr 3, 2025

Conversation

PIG208
Copy link
Member

@PIG208 PIG208 commented Apr 1, 2025

@PIG208 PIG208 changed the title Add API support for local_id/local_message_id and queue_id Add API support for local_id/local_message_id and queue_id (1/n) Apr 1, 2025
@PIG208 PIG208 changed the title Add API support for local_id/local_message_id and queue_id (1/n) locale echo (1/n): Add API support for local_id/local_message_id and queue_id Apr 1, 2025
@PIG208 PIG208 added the maintainer review PR ready for review by Zulip maintainers label Apr 1, 2025
@PIG208 PIG208 requested a review from rajveermalviya April 1, 2025 22:16
Copy link
Member

@rajveermalviya rajveermalviya 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 working on this! This LGTM.

(Also, for the PR title you probably meant 'local echo' instead of 'locale echo').

@rajveermalviya rajveermalviya requested a review from gnprice April 2, 2025 12:00
@rajveermalviya rajveermalviya added integration review Added by maintainers when PR may be ready for integration and removed maintainer review PR ready for review by Zulip maintainers labels Apr 2, 2025
@PIG208 PIG208 changed the title locale echo (1/n): Add API support for local_id/local_message_id and queue_id local echo (1/n): Add API support for local_id/local_message_id and queue_id Apr 2, 2025
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! Comments below.

Comment on lines -668 to -669
// TODO use [JsonSerializable] here too, using its customization features,
// in order to skip the boilerplate in [fromJson] and [toJson].
Copy link
Member

Choose a reason for hiding this comment

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

Cool, nice to have taken care of this.

@@ -665,8 +665,7 @@ class UserTopicEvent extends Event {
}

/// A Zulip event of type `message`: https://zulip.com/api/get-events#message
// TODO use [JsonSerializable] here too, using its customization features,
// in order to skip the boilerplate in [fromJson] and [toJson].
@JsonSerializable(fieldRename: FieldRename.snake)
Copy link
Member

Choose a reason for hiding this comment

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

commit-message nits:

It will make it easier to add addditional fields on the event later

needs period (commit-message bodies use complete sentences); and s/ddd/dd/

@@ -1,6 +1,5 @@
import 'package:checks/checks.dart';
import 'package:test/scaffolding.dart';
import 'package:zulip/api/model/events.dart';
Copy link
Member

Choose a reason for hiding this comment

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

commit-message nit:

A neat side-effect is that we can cut some imports from lib/api/model/.

s/from/of/

I parsed "cut some imports from $file" as "cut X from Y", meaning that Y had some X which got cut. Then was puzzled why an example_data change would affect that file, then saw the file indeed wasn't touched, then realized what you meant 🙂

@@ -367,8 +367,7 @@ void main() {
List.generate(30, (i) => eg.streamMessage(stream: stream)));

check(model).messages.length.equals(30);
await store.handleEvent(MessageEvent(id: 0,
message: eg.streamMessage(stream: stream)));
await store.handleEvent(eg.messageEvent(eg.streamMessage(stream: stream)));
Copy link
Member

Choose a reason for hiding this comment

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

Let's use the handy helper that simplifies these:

Suggested change
await store.handleEvent(eg.messageEvent(eg.streamMessage(stream: stream)));
await store.addMessage(eg.streamMessage(stream: stream));

In fact maybe switch to that in a preceding prep commit, and then the bulk of these don't need to change in the eg.messageEvent commit.

'queue_id': '"abc:123"',
'local_id': '"456"',
'queue_id': 'abc:123',
Copy link
Member

Choose a reason for hiding this comment

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

Hmm, is this really NFC?

api [nfc]: Use RawParameter when passing queue_id

Seems like it changes what we'll actually send to the server.

I guess it's NFC for the actual app, in that we don't currently pass this parameter in calls to sendMessage. Still I'd call this just api:, not api [nfc]:, because it's definitely a functional change to the API bindings.

Copy link
Member

Choose a reason for hiding this comment

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

Strings in general should not be JSON-encoded in Zulip APIs.

It's true that a string as a top-level parameter to a Zulip API endpoint is typically sent as itself, not JSON-encoded. That isn't always true, though. Right in this same endpoint, when passing a channel name to to, that's expected to be as JSON and not a raw string.

(The server does currently accept a raw stream name there, apparently, but that isn't and fundamentally can't be reliable.)

Moreover our docs don't really cover where this is the case or isn't, except via the curl examples; and the curl example for this endpoint doesn't have queue_id.

So for this change, I think the key point is that you've tested and the raw string is what works.

Comment on lines 685 to 690
// The format of this is documented to be chosen freely by the client.
//
// When present, this corresponds to the JSON-encoded int, "local_id",
// from a previous [sendMessage] call by us.
@JsonKey(fromJson: _localMessageIdFromJson, toJson: _localMessageIdToJson)
final int? localMessageId;
Copy link
Member

Choose a reason for hiding this comment

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

Since this isn't yet covered by the API docs (which are linked from this class's dartdoc), let's include dartdoc with a link to the best reference we have. That'd be the link in the commit message:
https://chat.zulip.org/#narrow/channel/412-api-documentation/topic/local_id.2C.20queue_id.2Fsender_queue_id/near/2135340

Comment on lines 197 to 200
// This is documented to be an optional String whose format is chosen freely
// by the client. Use a JSON-encoded int consistently, so that we know how
// to decode it when we receive the corresponding "message" event.
// See also: [MessageEvent.localMessageId]
Copy link
Member

Choose a reason for hiding this comment

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

In general I'd rather keep these bindings as close as possible to the underlying Zulip API — adapting the API to reasonable Dart idioms, but generally not encoding our particular choices of how the app uses the API.

One reason is that it saves us from needing this sort of comment to explain how and why the internal API differs from the one that's in the API docs 🙂 (or in this case hopefully will be documented there).

What if we just leave this as a string? The MessageStore code that calls this endpoint, and then consumes the local_message_id in the resulting MessageEvent, will end up converting an int message ID to string and back, but that's fine — it's really that code that is responsible for matching those requests and events up with each other, so that's a natural place to put the responsibility for deciding what these strings should look like.

Copy link
Member Author

Choose a reason for hiding this comment

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

Yeah, I agree it will be cleaner when we handle that encoding at a different layer. In this case it should be self-contained in MessageStore.

@@ -620,6 +620,9 @@ UserTopicEvent userTopicEvent(
);
}

MessageEvent messageEvent(Message message) =>
Copy link
Member

Choose a reason for hiding this comment

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

commit-message nit:

example_data [nfc]: Add messageEvent

Better:

test [nfc]: Add eg.messageEvent

For examples of previous summary lines, try this command:

$ git log --format='%>(20)%an %cs %h %s' --grep 'eg\.|example' origin \
    | grep -P '\bexample|\beg\b'

Copy link
Member Author

Choose a reason for hiding this comment

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

Thanks! I added a -E flag to the git log command to get this working.

Copy link
Member

Choose a reason for hiding this comment

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

Ah yeah, thanks. I have this in my ~/.gitconfig, which I recommend:

[grep]
        # "Basic" regexps are a relic.  Extended would be fine; Perl is better.
        patternType = perl

That's equivalent to a -P flag, which goes farther in the same direction as -E.

@PIG208
Copy link
Member Author

PIG208 commented Apr 2, 2025

Updated the PR. This should be ready for review. Thanks!

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! This looks good; just small comments below.

Comment on lines 685 to 686
// The format of this is documented to be chosen freely by the client.
//
// When present, this corresponds to the JSON-encoded int, "local_id",
// from a previous [sendMessage] call by us.
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
// The format of this is documented to be chosen freely by the client.
//
// When present, this corresponds to the JSON-encoded int, "local_id",
// from a previous [sendMessage] call by us.
// When present, this equals the "local_id" parameter
// from a previous [sendMessage] call by us.

The details about what we choose to put into "local_id" are no longer something this needs in order to justify how this code works, so this can stick to documenting the API itself.

Comment on lines -196 to +197
if (queueId != null) 'queue_id': queueId, // TODO should this use RawParameter?
if (localId != null) 'local_id': localId, // TODO should this use RawParameter?
if (queueId != null) 'queue_id': RawParameter(queueId),
if (localId != null) 'local_id': RawParameter(localId),
Copy link
Member

Choose a reason for hiding this comment

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

The change to localId here on this route is now totally parallel to the change to queueId, so probably best done in the same commit.

Then the change adding localMessageId on the event can remain in its own subsequent commit.

PIG208 added 3 commits April 3, 2025 13:53
It will make it easier to add additional fields on the event later.
This leaves some tests that handle MessageEvent untouched,
as they conceptually aren't just setting up the store, and the helper
adds a layer of indirection (or they simply don't have access to a
PerAccountStore).
MessageEvent constructor calls are generally quite short, and there are
no boring required fields other than an integer id.  However, as we add
localMessageId to the event, it would be expected to have a boring
value most of the time, so prepare for that.

A neat side-effect is that we can cut some imports of lib/api/model/.

(Some places that use non-zero event IDs now use 0, but we don't
care much about this value anyway.)
@PIG208
Copy link
Member Author

PIG208 commented Apr 3, 2025

Thanks! Updated the PR.

@gnprice
Copy link
Member

gnprice commented Apr 3, 2025

Thanks! A commit message needs updating:

api: Use RawParameter when passing queue_id

This endpoint in particular expects queue_id to be a String just
passed as itself, not JSON-encoded.

We will handle local_id separately because that's a bit different.

as that commit now handles local_id too :-)

Otherwise looks all ready.

PIG208 added 2 commits April 3, 2025 14:54
This endpoint in particular expects queue_id and local_id each to
be a String just passed as itself, not JSON-encoded.
This will be used for local echoing.  local_message_id on message
event is not yet fully documented; see CZO discussion:
  https://chat.zulip.org/#narrow/channel/412-api-documentation/topic/local_id.2C.20queue_id.2Fsender_queue_id/near/2135340
@PIG208
Copy link
Member Author

PIG208 commented Apr 3, 2025

Oops. Just pushed another update.

@gnprice
Copy link
Member

gnprice commented Apr 3, 2025

Thanks! Looks good; merging.

@gnprice gnprice merged commit d224a5b into zulip:main Apr 3, 2025
1 check passed
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.

3 participants