-
Notifications
You must be signed in to change notification settings - Fork 5
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: add providers endpoint and enable muxing #253
Conversation
Pull Request Test Coverage Report for Build 13156904333Details
💛 - Coveralls |
import { useQuery } from "@tanstack/react-query"; | ||
import { v1GetProviderEndpointOptions } from "@/api/generated/@tanstack/react-query.gen"; | ||
import { AddProviderEndpointRequest, ProviderType } from "@/api/generated"; | ||
import { useEffect, useState } from "react"; | ||
|
||
export function useProvider(providerId: string) { | ||
const [provider, setProvider] = useState<AddProviderEndpointRequest>({ | ||
name: "", | ||
description: "", | ||
auth_type: null, | ||
provider_type: ProviderType.OPENAI, | ||
endpoint: "", | ||
api_key: "", | ||
}); | ||
|
||
const { data, isPending, isError } = useQuery({ | ||
...v1GetProviderEndpointOptions({ path: { provider_id: providerId } }), | ||
}); | ||
|
||
useEffect(() => { | ||
if (data) { | ||
setProvider(data); | ||
} | ||
}, [data]); | ||
|
||
return { isPending, isError, provider, setProvider }; | ||
} |
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.
suggestion: eliminate the useState+useEffect & lean on react-query for state management?
import { useQuery } from "@tanstack/react-query"; | |
import { v1GetProviderEndpointOptions } from "@/api/generated/@tanstack/react-query.gen"; | |
import { AddProviderEndpointRequest, ProviderType } from "@/api/generated"; | |
import { useEffect, useState } from "react"; | |
export function useProvider(providerId: string) { | |
const [provider, setProvider] = useState<AddProviderEndpointRequest>({ | |
name: "", | |
description: "", | |
auth_type: null, | |
provider_type: ProviderType.OPENAI, | |
endpoint: "", | |
api_key: "", | |
}); | |
const { data, isPending, isError } = useQuery({ | |
...v1GetProviderEndpointOptions({ path: { provider_id: providerId } }), | |
}); | |
useEffect(() => { | |
if (data) { | |
setProvider(data); | |
} | |
}, [data]); | |
return { isPending, isError, provider, setProvider }; | |
} | |
import { useQuery } from "@tanstack/react-query"; | |
import { v1GetProviderEndpointOptions } from "@/api/generated/@tanstack/react-query.gen"; | |
import { ProviderType, V1GetProviderEndpointResponse } from "@/api/generated"; | |
const DEFAULT_PROVIDER = { | |
name: "", | |
description: "", | |
auth_type: null, | |
provider_type: ProviderType.OPENAI, | |
endpoint: "", | |
api_key: "", | |
} as const satisfies V1GetProviderEndpointResponse & { api_key: string }; | |
export function useProvider(providerId: string) { | |
const { data = DEFAULT_PROVIDER, ...rest } = useQuery({ | |
...v1GetProviderEndpointOptions({ path: { provider_id: providerId } }), | |
}); | |
return { data, ...rest }; | |
} | |
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.
that's cool for default state, but how I can handle the setProvider? and submit the data within a mutation? I guess that having the Form the state is needed 🤔
export const usePreferredModelWorkspace = (workspaceName: string) => { | ||
const options: V1GetWorkspaceMuxesData & | ||
Omit<V1GetWorkspaceMuxesData, "body"> = useMemo( | ||
() => ({ | ||
path: { workspace_name: workspaceName }, | ||
}), | ||
[workspaceName], | ||
); | ||
const { data } = usePreferredModel(options); | ||
const { preferredModel, setPreferredModel } = useModelValue(); | ||
|
||
useEffect(() => { | ||
const providerModel = data?.[0]; | ||
if (providerModel) { | ||
setPreferredModel(providerModel); | ||
} | ||
}, [data, setPreferredModel]); | ||
|
||
return { preferredModel, setPreferredModel }; | ||
}; |
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.
nit: I would prefer to do this without zustand. We are already doing something similar with useActiveWorkspaceName
:
import { ListActiveWorkspacesResponse } from "@/api/generated";
import { useActiveWorkspaces } from "./use-active-workspaces";
// NOTE: This needs to be a stable function reference to enable memo-isation of
// the select operation on each React re-render.
function select(data: ListActiveWorkspacesResponse | undefined): string | null {
return data?.workspaces?.[0]?.name ?? null;
}
export function useActiveWorkspaceName() {
return useActiveWorkspaces({
select,
});
}
The reason I'm opposed to zustand is the data is already globally available via react-query, but we're introducing a copy/syncing mechanism that seems unnecessary.
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.
the zustand was a leftover from the muxing first PR with the array and drag n drop feature, but I guess that having the Form the state is needed 🤔 so I would replace it with a basic useState
Add provider endpoint page CRUD, this allow to select preferred model per workspace (muxing).
Add providers item into Settings menu

Create provider and save preferred model in the workspace
Kapture.2025-02-05.at.10.52.35.mp4
Delete provider
Kapture.2025-02-05.at.10.14.46.mp4
nit: tests are on going