Files
yaak/src-web/hooks/useGrpc.ts
Gregory Schier e27ed9becc bidi hacked!
2024-02-02 01:10:54 -08:00

125 lines
3.6 KiB
TypeScript

import { useMutation, useQuery } from '@tanstack/react-query';
import { invoke } from '@tauri-apps/api';
import { message } from '@tauri-apps/api/dialog';
import { emit } from '@tauri-apps/api/event';
import { useState } from 'react';
import { send } from 'vite';
import { useListenToTauriEvent } from './useListenToTauriEvent';
interface ReflectResponseService {
name: string;
methods: { name: string; schema: string; serverStreaming: boolean; clientStreaming: boolean }[];
}
export interface GrpcMessage {
message: string;
time: Date;
type: 'server' | 'client' | 'info';
}
export function useGrpc(url: string | null) {
const [messages, setMessages] = useState<GrpcMessage[]>([]);
const [activeConnectionId, setActiveConnectionId] = useState<string | null>(null);
useListenToTauriEvent<string>(
'grpc_message',
(event) => {
setMessages((prev) => [
...prev,
{ message: event.payload, time: new Date(), type: 'server' },
]);
},
[],
);
const unary = useMutation<string, string, { service: string; method: string; message: string }>({
mutationKey: ['grpc_unary', url],
mutationFn: async ({ service, method, message }) => {
if (url === null) throw new Error('No URL provided');
return (await invoke('cmd_grpc_call_unary', {
endpoint: url,
service,
method,
message,
})) as string;
},
});
const serverStreaming = useMutation<
void,
string,
{ service: string; method: string; message: string }
>({
mutationKey: ['grpc_server_streaming', url],
mutationFn: async ({ service, method, message }) => {
if (url === null) throw new Error('No URL provided');
setMessages([
{ type: 'client', message: JSON.stringify(JSON.parse(message)), time: new Date() },
]);
const id: string = await invoke('cmd_grpc_server_streaming', {
endpoint: url,
service,
method,
message,
});
setActiveConnectionId(id);
},
});
const bidiStreaming = useMutation<
void,
string,
{ service: string; method: string; message: string }
>({
mutationKey: ['grpc_bidi_streaming', url],
mutationFn: async ({ service, method, message }) => {
if (url === null) throw new Error('No URL provided');
const id: string = await invoke('cmd_grpc_bidi_streaming', {
endpoint: url,
service,
method,
message,
});
setMessages([{ type: 'info', message: `Started connection ${id}`, time: new Date() }]);
setActiveConnectionId(id);
},
});
const send = useMutation({
mutationKey: ['grpc_send', url],
mutationFn: async ({ message }: { message: string }) => {
await emit('grpc_message_in', { Message: message });
setMessages((m) => [...m, { type: 'client', message, time: new Date() }]);
},
});
const cancel = useMutation({
mutationKey: ['grpc_cancel', url],
mutationFn: async () => {
await emit('grpc_message_in', 'Cancel');
setActiveConnectionId(null);
setMessages((m) => [
...m,
{ type: 'info', message: 'Cancelled by client', time: new Date() },
]);
},
});
const reflect = useQuery<ReflectResponseService[]>({
queryKey: ['grpc_reflect', url ?? ''],
queryFn: async () => {
if (url === null) return [];
return (await invoke('cmd_grpc_reflect', { endpoint: url })) as ReflectResponseService[];
},
});
return {
unary,
serverStreaming,
bidiStreaming,
schema: reflect.data,
cancel,
messages,
isStreaming: activeConnectionId !== null,
send,
};
}