summaryrefslogtreecommitdiffstats
path: root/src/hooks/useRequests.tsx
blob: 2b8393e83cc48bfd486647b4cf67e3a9b1026cbb (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
import {useCallback, useEffect, useMemo, useState} from "react";
import {getHost} from "../utils";

type Dict = {
  [key: string]: string
}

export interface RequestPayload {
  id: string
  body: string
  cookies: Dict
  headers: Dict
  method: 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH'
  path: string
}

interface Request {
  type: 'request',
  payload: RequestPayload,
}

export interface ResponsePayload {
  id: string
  timing: number
  body: string
  cookies: Dict
  headers: Dict
  status: number
}

interface Response {
  type: 'response',
  payload: ResponsePayload,
}

interface Historic {
  type: 'historic'
  payload: (Request | Response)[]
}

export interface RequestResponse {
  request: RequestPayload
  response?: ResponsePayload
}

export enum ReadyState {
  CONNECTING = 0,
  OPEN = 1,
  CLOSING = 2,
  CLOSED = 3,
}

export interface useRequestsProps {
  onConnect: () => Promise<void>
}

export interface UseRequests {
  calls: RequestResponse[]
  readyState: ReadyState
}

export default function useRequests({ onConnect }: useRequestsProps): UseRequests {
  const wsHost = useMemo(getHost, []);

  const [requests, setRequests] = useState<RequestPayload[]>([]);
  const [responses, setResponses] = useState<ResponsePayload[]>([]);


  const connect = useCallback(() => (
    new WebSocket(`ws://${wsHost}/inspect/`)
  ), [wsHost]);

  const [ws, setWs] = useState<WebSocket>(() => connect());
  const [readyState, setReadyState] = useState<ReadyState>(ws.readyState);

  useEffect(() => {
    setReadyState(ws.readyState);

    const onClose = () => {
      setReadyState(ws.readyState);
      setWs(connect());
    }
    const onOpen = () => {
      onConnect();
      setReadyState(ws.readyState);
    }
    const onMessage = ({ data }) => {
      const { type, payload } = JSON.parse(data) as Historic | Request | Response

      switch (type) {
        case 'historic':
          const requests = (payload as (Request | Response)[]).filter(({ type }) => type === 'request');
          const responses = (payload as (Request | Response)[]).filter(({ type }) => type === 'response');
          setRequests((rqs) => [...rqs, ...requests.map(({ payload }) => payload as RequestPayload)]);
          setResponses((rps) => [...rps, ...responses.map(({ payload }) => payload as ResponsePayload)]);
          break
        case 'request':
          setRequests((rqs) => [...rqs, payload as RequestPayload])
          break
        case 'response':
          setResponses((rps) => [...rps, payload as ResponsePayload])
          break
      }
    }

    ws.addEventListener('message', onMessage)
    ws.addEventListener('close', onClose)
    ws.addEventListener('open', onOpen)

    return () => {
      ws.removeEventListener('message', onMessage);
      ws.removeEventListener('close', onClose);
      ws.removeEventListener('open', onOpen)
    }
  }, [ws])

  return {
    calls: useMemo<RequestResponse[]>(() => requests.map((request) => ({
      request: request,
      response: responses.find(({id}) => id === request.id)
    })), [requests, responses]),
    readyState,
  }
}