feat(coderd): add pagination to chat list endpoint (#22887)

Adds offset and cursor-based pagination to the `GET
/api/experimental/chats` endpoint, following the exact same patterns
used by `GetUsers` and `GetTemplateVersionsByTemplateID`.

## Changes

### Database
- Add `after_id`, `offset_opt`, `limit_opt` params to
`GetChatsByOwnerID` SQL query
- Use composite `(updated_at, id) DESC` cursor for stable, deterministic
pagination
- Add migration with composite index on `chats (owner_id, updated_at
DESC, id DESC)`

### Backend
- Use `ParsePagination()` in `listChats` handler (matches `users.go`
pattern)
- Add `Pagination` field to `ListChatsOptions` SDK struct

### Frontend
- Add `infiniteChats()` query factory using `useInfiniteQuery` with
offset-based page params (same pattern as `infiniteWorkspaceBuilds`)
- Update `AgentsPage` to use `useInfiniteQuery`
- Add "Show more" button at the bottom of the agents sidebar (matches
`HistorySidebar` pattern)
- Keep existing `chats()` query for non-paginated uses (e.g., parent
chat lookup in `AgentDetail`)

### Tests
- Add `TestListChats/Pagination` covering `limit`, `after_id` cursor,
`offset`, and no-limit behavior
This commit is contained in:
Kyle Carberry
2026-03-10 06:55:33 -07:00
committed by GitHub
parent beed379b1d
commit e18ce505ec
14 changed files with 244 additions and 19 deletions
+10
View File
@@ -179,8 +179,18 @@ func (api *API) listChats(rw http.ResponseWriter, r *http.Request) {
ctx := r.Context()
apiKey := httpmw.APIKey(r)
paginationParams, ok := ParsePagination(rw, r)
if !ok {
return
}
params := database.GetChatsByOwnerIDParams{
OwnerID: apiKey.UserID,
AfterID: paginationParams.AfterID,
// #nosec G115 - Pagination offsets are small and fit in int32
OffsetOpt: int32(paginationParams.Offset),
// #nosec G115 - Pagination limits are small and fit in int32
LimitOpt: int32(paginationParams.Limit),
}
if v := r.URL.Query().Get("archived"); v != "" {
b, err := strconv.ParseBool(v)
+84
View File
@@ -387,6 +387,90 @@ func TestListChats(t *testing.T) {
_, err := unauthenticatedClient.ListChats(ctx, nil)
requireSDKError(t, err, http.StatusUnauthorized)
})
t.Run("Pagination", func(t *testing.T) {
t.Parallel()
ctx := testutil.Context(t, testutil.WaitLong)
client, _ := newChatClientWithDatabase(t)
_ = coderdtest.CreateFirstUser(t, client)
_ = createChatModelConfig(t, client)
// Create 5 chats.
const totalChats = 5
createdChats := make([]codersdk.Chat, 0, totalChats)
for i := 0; i < totalChats; i++ {
chat, err := client.CreateChat(ctx, codersdk.CreateChatRequest{
Content: []codersdk.ChatInputPart{
{
Type: codersdk.ChatInputPartTypeText,
Text: fmt.Sprintf("chat-%d", i),
},
},
})
require.NoError(t, err)
createdChats = append(createdChats, chat)
}
// Fetch first page with limit=2.
page1, err := client.ListChats(ctx, &codersdk.ListChatsOptions{
Pagination: codersdk.Pagination{Limit: 2},
})
require.NoError(t, err)
require.Len(t, page1, 2)
// Fetch second page using after_id from last item of page 1.
page2, err := client.ListChats(ctx, &codersdk.ListChatsOptions{
Pagination: codersdk.Pagination{
AfterID: uuid.MustParse(page1[len(page1)-1].ID.String()),
Limit: 2,
},
})
require.NoError(t, err)
require.Len(t, page2, 2)
// Ensure page1 and page2 have no overlap.
page1IDs := make(map[uuid.UUID]struct{})
for _, c := range page1 {
page1IDs[c.ID] = struct{}{}
}
for _, c := range page2 {
_, overlap := page1IDs[c.ID]
require.False(t, overlap, "page2 should not contain items from page1")
}
// Fetch third page — should have 1 remaining chat.
page3, err := client.ListChats(ctx, &codersdk.ListChatsOptions{
Pagination: codersdk.Pagination{
AfterID: uuid.MustParse(page2[len(page2)-1].ID.String()),
Limit: 2,
},
})
require.NoError(t, err)
require.Len(t, page3, 1)
// All 5 chats should be accounted for.
allIDs := make(map[uuid.UUID]struct{})
for _, c := range append(append(page1, page2...), page3...) {
allIDs[c.ID] = struct{}{}
}
for _, c := range createdChats {
_, found := allIDs[c.ID]
require.True(t, found, "chat %s should appear in paginated results", c.ID)
}
// Fetch with offset=3, limit=2 — should return 2 chats.
offsetPage, err := client.ListChats(ctx, &codersdk.ListChatsOptions{
Pagination: codersdk.Pagination{Offset: 3, Limit: 2},
})
require.NoError(t, err)
require.Len(t, offsetPage, 2)
// No limit should return all chats.
allChats, err := client.ListChats(ctx, nil)
require.NoError(t, err)
require.Len(t, allChats, totalChats)
})
}
func TestListChatModels(t *testing.T) {
+2
View File
@@ -3534,6 +3534,8 @@ CREATE INDEX idx_chats_last_model_config_id ON chats USING btree (last_model_con
CREATE INDEX idx_chats_owner ON chats USING btree (owner_id);
CREATE INDEX idx_chats_owner_updated_id ON chats USING btree (owner_id, updated_at DESC, id DESC);
CREATE INDEX idx_chats_parent_chat_id ON chats USING btree (parent_chat_id);
CREATE INDEX idx_chats_pending ON chats USING btree (status) WHERE (status = 'pending'::chat_status);
@@ -0,0 +1 @@
DROP INDEX IF EXISTS idx_chats_owner_updated_id;
@@ -0,0 +1 @@
CREATE INDEX idx_chats_owner_updated_id ON chats (owner_id, updated_at DESC, id DESC);
+38 -4
View File
@@ -3454,17 +3454,51 @@ WHERE
WHEN $2 :: boolean IS NULL THEN true
ELSE chats.archived = $2 :: boolean
END
AND CASE
-- This allows using the last element on a page as effectively a cursor.
-- This is an important option for scripts that need to paginate without
-- duplicating or missing data.
WHEN $3 :: uuid != '00000000-0000-0000-0000-000000000000'::uuid THEN (
-- The pagination cursor is the last ID of the previous page.
-- The query is ordered by the updated_at field, so select all
-- rows before the cursor.
(updated_at, id) < (
SELECT
updated_at, id
FROM
chats
WHERE
id = $3
)
)
ELSE true
END
ORDER BY
updated_at DESC
-- Deterministic and consistent ordering of all rows, even if they share
-- a timestamp. This is to ensure consistent pagination.
(updated_at, id) DESC OFFSET $4
LIMIT
-- The chat list is unbounded and expected to grow large.
-- Default to 50 to prevent accidental excessively large queries.
COALESCE(NULLIF($5 :: int, 0), 50)
`
type GetChatsByOwnerIDParams struct {
OwnerID uuid.UUID `db:"owner_id" json:"owner_id"`
Archived sql.NullBool `db:"archived" json:"archived"`
OwnerID uuid.UUID `db:"owner_id" json:"owner_id"`
Archived sql.NullBool `db:"archived" json:"archived"`
AfterID uuid.UUID `db:"after_id" json:"after_id"`
OffsetOpt int32 `db:"offset_opt" json:"offset_opt"`
LimitOpt int32 `db:"limit_opt" json:"limit_opt"`
}
func (q *sqlQuerier) GetChatsByOwnerID(ctx context.Context, arg GetChatsByOwnerIDParams) ([]Chat, error) {
rows, err := q.db.QueryContext(ctx, getChatsByOwnerID, arg.OwnerID, arg.Archived)
rows, err := q.db.QueryContext(ctx, getChatsByOwnerID,
arg.OwnerID,
arg.Archived,
arg.AfterID,
arg.OffsetOpt,
arg.LimitOpt,
)
if err != nil {
return nil, err
}
+26 -1
View File
@@ -113,8 +113,33 @@ WHERE
WHEN sqlc.narg('archived') :: boolean IS NULL THEN true
ELSE chats.archived = sqlc.narg('archived') :: boolean
END
AND CASE
-- This allows using the last element on a page as effectively a cursor.
-- This is an important option for scripts that need to paginate without
-- duplicating or missing data.
WHEN @after_id :: uuid != '00000000-0000-0000-0000-000000000000'::uuid THEN (
-- The pagination cursor is the last ID of the previous page.
-- The query is ordered by the updated_at field, so select all
-- rows before the cursor.
(updated_at, id) < (
SELECT
updated_at, id
FROM
chats
WHERE
id = @after_id
)
)
ELSE true
END
ORDER BY
updated_at DESC;
-- Deterministic and consistent ordering of all rows, even if they share
-- a timestamp. This is to ensure consistent pagination.
(updated_at, id) DESC OFFSET @offset_opt
LIMIT
-- The chat list is unbounded and expected to grow large.
-- Default to 50 to prevent accidental excessively large queries.
COALESCE(NULLIF(@limit_opt :: int, 0), 50);
-- name: ListChildChatsByParentID :many
SELECT
+12 -5
View File
@@ -7,7 +7,6 @@ import (
"io"
"mime"
"net/http"
"net/url"
"strings"
"time"
@@ -540,15 +539,23 @@ type chatStreamEnvelope struct {
// ListChatsOptions are optional parameters for ListChats.
type ListChatsOptions struct {
Archived *bool
Pagination
}
// ListChats returns all chats for the authenticated user.
func (c *Client) ListChats(ctx context.Context, opts *ListChatsOptions) ([]Chat, error) {
qp := url.Values{}
if opts != nil && opts.Archived != nil {
qp.Set("archived", fmt.Sprintf("%t", *opts.Archived))
var reqOpts []RequestOption
if opts != nil {
reqOpts = append(reqOpts, opts.Pagination.asRequestOption())
if opts.Archived != nil {
reqOpts = append(reqOpts, func(r *http.Request) {
q := r.URL.Query()
q.Set("archived", fmt.Sprintf("%t", *opts.Archived))
r.URL.RawQuery = q.Encode()
})
}
}
res, err := c.Request(ctx, http.MethodGet, fmt.Sprintf("/api/experimental/chats?%s", qp.Encode()), nil)
res, err := c.Request(ctx, http.MethodGet, "/api/experimental/chats", nil, reqOpts...)
if err != nil {
return nil, err
}
+7 -3
View File
@@ -2937,13 +2937,17 @@ class ApiMethods {
};
// Chat API methods
getChats = async (): Promise<TypesGen.Chat[]> => {
getChats = async (req?: {
after_id?: string;
limit?: number;
offset?: number;
archived?: string;
}): Promise<TypesGen.Chat[]> => {
const response = await this.axios.get<TypesGen.Chat[]>(
"/api/experimental/chats",
getURLWithSearchParams("/api/experimental/chats", req),
);
return response.data;
};
getChat = async (chatId: string): Promise<TypesGen.ChatWithMessages> => {
const response = await this.axios.get<TypesGen.ChatWithMessages>(
`/api/experimental/chats/${chatId}`,
+29 -1
View File
@@ -1,10 +1,38 @@
import { API, type ChatDiffStatusResponse } from "api/api";
import type * as TypesGen from "api/typesGenerated";
import type { QueryClient } from "react-query";
import type { QueryClient, UseInfiniteQueryOptions } from "react-query";
export const chatsKey = ["chats"] as const;
export const chatKey = (chatId: string) => ["chats", chatId] as const;
const DEFAULT_CHAT_PAGE_LIMIT = 50;
export const infiniteChats = (opts?: { archived?: boolean }) => {
const limit = DEFAULT_CHAT_PAGE_LIMIT;
return {
queryKey: [...chatsKey, opts],
getNextPageParam: (lastPage: TypesGen.Chat[], pages: TypesGen.Chat[][]) => {
if (lastPage.length < limit) {
return undefined;
}
return pages.length + 1;
},
initialPageParam: 0,
queryFn: ({ pageParam }: { pageParam: unknown }) => {
if (typeof pageParam !== "number") {
throw new Error("pageParam must be a number");
}
return API.getChats({
limit,
offset: pageParam <= 0 ? 0 : (pageParam - 1) * limit,
archived: opts?.archived?.toString(),
});
},
refetchOnWindowFocus: true as const,
} satisfies UseInfiniteQueryOptions<TypesGen.Chat[]>;
};
export const chats = () => ({
queryKey: chatsKey,
queryFn: () => API.getChats(),
+1 -1
View File
@@ -3260,7 +3260,7 @@ export interface LinkConfig {
/**
* ListChatsOptions are optional parameters for ListChats.
*/
export interface ListChatsOptions {
export interface ListChatsOptions extends Pagination {
readonly Archived: boolean | null;
}
+11 -4
View File
@@ -8,9 +8,9 @@ import {
chatModelConfigs,
chatModels,
chatSystemPrompt,
chats,
chatsKey,
createChat,
infiniteChats,
unarchiveChat,
updateChatSystemPrompt,
} from "api/queries/chats";
@@ -40,7 +40,12 @@ import {
useRef,
useState,
} from "react";
import { useMutation, useQuery, useQueryClient } from "react-query";
import {
useInfiniteQuery,
useMutation,
useQuery,
useQueryClient,
} from "react-query";
import { useNavigate, useParams } from "react-router";
import { toast } from "sonner";
import { createReconnectingWebSocket } from "utils/reconnectingWebSocket";
@@ -144,7 +149,7 @@ const AgentsPage: FC = () => {
};
}, []);
const chatsQuery = useQuery(chats());
const chatsQuery = useInfiniteQuery(infiniteChats());
const chatModelsQuery = useQuery(chatModels());
const chatModelConfigsQuery = useQuery(chatModelConfigs());
const createMutation = useMutation(createChat(queryClient));
@@ -247,7 +252,7 @@ const AgentsPage: FC = () => {
return next;
});
}, []);
const chatList = chatsQuery.data ?? [];
const chatList = chatsQuery.data?.pages.flat() ?? [];
const isArchiving =
archiveAgentMutation.isPending || archiveAndDeleteMutation.isPending;
const archivingChatId =
@@ -517,6 +522,8 @@ const AgentsPage: FC = () => {
isModelConfigsLoading={chatModelConfigsQuery.isLoading}
modelCatalogError={chatModelsQuery.error}
isAgentsAdmin={isAgentsAdmin}
hasNextPage={chatsQuery.hasNextPage}
onLoadMore={() => void chatsQuery.fetchNextPage()}
/>
);
};
@@ -53,6 +53,8 @@ interface AgentsPageViewProps {
isModelCatalogLoading: boolean;
isModelConfigsLoading: boolean;
modelCatalogError: unknown;
hasNextPage: boolean | undefined;
onLoadMore: () => void;
}
export const AgentsPageView: FC<AgentsPageViewProps> = ({
@@ -79,6 +81,8 @@ export const AgentsPageView: FC<AgentsPageViewProps> = ({
isModelCatalogLoading,
isModelConfigsLoading,
modelCatalogError,
hasNextPage,
onLoadMore,
}) => {
const {
chatErrorReasons,
@@ -116,6 +120,8 @@ export const AgentsPageView: FC<AgentsPageViewProps> = ({
isLoading={isChatsLoading}
loadError={chatsLoadError}
onRetryLoad={onRetryChatsLoad}
hasNextPage={hasNextPage}
onLoadMore={onLoadMore}
onCollapse={onCollapseSidebar}
/>
</div>
@@ -72,6 +72,8 @@ interface AgentsSidebarProps {
isLoading?: boolean;
loadError?: unknown;
onRetryLoad?: () => void;
hasNextPage?: boolean;
onLoadMore?: () => void;
onCollapse?: () => void;
}
@@ -537,6 +539,8 @@ export const AgentsSidebar: FC<AgentsSidebarProps> = (props) => {
isLoading = false,
loadError,
onRetryLoad,
hasNextPage,
onLoadMore,
onCollapse,
} = props;
const { agentId, chatId } = useParams<{
@@ -793,6 +797,18 @@ export const AgentsSidebar: FC<AgentsSidebarProps> = (props) => {
)}
</div>
)}
{hasNextPage && (
<div className="px-2 py-2">
<Button
size="sm"
variant="outline"
className="w-full"
onClick={onLoadMore}
>
Show more
</Button>
</div>
)}
</ChatTreeContext.Provider>
)}
</div>