Message Search

Search messages across channels using full-text search or specific field filters. Enable or disable search indexing per channel type in the Stream Dashboard.

Searching Messages

Search requires a channel filter and either a text query or message filter conditions.

require 'getstream_ruby'
Models = GetStream::Generated::Models

page1 = client.chat.search(Models::SearchPayload.new(
  filter_conditions: { 'cid' => 'messaging:my-channel' },
  message_filter_conditions: { 'text' => { '$autocomplete' => 'supercali' } },
  sort: [
    Models::SortParamRequest.new(field: 'relevance', direction: -1),
    Models::SortParamRequest.new(field: 'updated_at', direction: 1)
  ],
  limit: 10
))

Query Parameters

NameTypeDescriptionDefaultOptional
filter_conditionsobjectChannel filters. Maximum 500 channels are searched. See Querying Channels.-
message_filter_conditionsobjectMessage filters. See Message Filter Conditions below. Either this or query is required.-
querystringFull-text search string. Equivalent to {text: {$q: <query>}}. Either this or message_filter_conditions is required.-
limitintegerNumber of messages to return.100
offsetintegerPagination offset. Cannot be used with sort or next.0
sortobject/arraySort order. Use field names with 1 (ascending) or -1 (descending).[{relevance: -1}, {id: 1}]
nextstringPagination cursor. See Pagination below.-

Message Filter Conditions

FieldDescriptionOperators
idMessage ID$eq, $gt, $gte, $lt, $lte, $in
textMessage text$q, $autocomplete, $eq, $gt, $gte, $lt, $lte, $in
typeMessage type. System and deleted messages are excluded$eq, $gt, $gte, $lt, $lte, $in
parent_idParent message ID (for replies)$eq, $gt, $gte, $lt, $lte, $in
reply_countNumber of replies$eq, $gt, $gte, $lt, $lte, $in
attachmentsWhether message has attachments$exists, $eq, $gt, $gte, $lt, $lte, $in
attachments.typeAttachment type$eq, $in
mentioned_users.idMentioned user ID$contains
user.idSender user ID$eq, $gt, $gte, $lt, $lte, $in
created_atCreation timestamp$eq, $gt, $gte, $lt, $lte, $in
updated_atUpdate timestamp$eq, $gt, $gte, $lt, $lte, $in
pinnedWhether message is pinned$eq
custom fieldAny custom field on the message$eq, $gt, $gte, $lt, $lte, $in

Sorting

Results are sorted by relevance by default, with message ID as a tiebreaker. If your query does not use $q or $autocomplete, all results are equally relevant.

Sort by any filterable field, including custom fields. Numeric custom fields are sorted numerically; string fields are sorted lexicographically.

Pagination

Two pagination methods are available:

Offset pagination allows access to up to 1,000 results. Results are sorted by relevance and message ID. You cannot use custom sorting with offset pagination.

Cursor pagination (using next/previous) allows access to all matching results with custom sorting. The response includes next and previous cursors to navigate between pages.

require 'getstream_ruby'
Models = GetStream::Generated::Models

channel_filters = { 'cid' => 'messaging:my-channel' }
message_filters = { 'text' => { '$autocomplete' => 'supercali' } }
sort = [
  Models::SortParamRequest.new(field: 'relevance', direction: -1),
  Models::SortParamRequest.new(field: 'updated_at', direction: 1),
  Models::SortParamRequest.new(field: 'my_custom_field', direction: -1)
]

# First page
page1 = client.chat.search(Models::SearchPayload.new(
  filter_conditions: channel_filters,
  message_filter_conditions: message_filters,
  sort: sort,
  limit: 10
))

# Next page
page2 = client.chat.search(Models::SearchPayload.new(
  filter_conditions: channel_filters,
  message_filter_conditions: message_filters,
  limit: 10,
  next: page1.next
))

# Previous page
page1_again = client.chat.search(Models::SearchPayload.new(
  filter_conditions: channel_filters,
  message_filter_conditions: message_filters,
  limit: 10,
  next: page2.previous
))