-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathQuery.re
120 lines (98 loc) · 2.63 KB
/
Query.re
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
open Revery.UI.React;
include S;
module type Query = {
type t('responseType) = 'responseType;
type status('responseType) =
| Idle
| Loading
| Error
| Data(t('responseType));
let useQuery:
(
~variables: Yojson.Basic.t=?,
(Yojson.Basic.t => t('responseType), string, 'b),
unit,
Hooks.t(
(
Hooks.Reducer.t(status('responseType)),
Hooks.Effect.t(option(Yojson.Basic.t))
) =>
'c,
'd,
)
) =>
(status('responseType), Hooks.t('c, 'd));
};
module Make = (C: BaseConfig) : Query => {
let baseUrl = C.baseUrl;
let headers = C.headers;
type t('responseType) = 'responseType;
type status('responseType) =
| Idle
| Loading
| Error
| Data(t('responseType));
type action('responseType) =
| FetchData
| Error
| Data('responseType);
let reducer = (action, _state): status('responseType) =>
switch (action) {
| FetchData => Loading
| Error => Error
| Data(obj) => Data(obj)
};
let initialState: status('responseType) = Idle;
let createBody = (~variables=`Assoc([]), ~query) => {
let payload =
`Assoc([("query", `String(query)), ("variables", variables)])
|> Yojson.Basic.to_string;
(payload, variables);
};
let useQuery = (~variables: option(Yojson.Basic.t)=?, definition, ()) => {
let (parse, graphqlQuery, composeVariables) = definition;
let%hook (state, dispatch) = Hooks.reducer(~initialState, reducer);
let subscribeToStore = query =>
Store.subscribe(
~query,
graphqlJson => {
let data =
graphqlJson
|> Yojson.Basic.from_string
|> Yojson.Basic.Util.member("data")
|> parse;
dispatch(Data(data));
},
);
let executeRequest = (query, variables) => {
dispatch(FetchData);
Fetch.post(
~body=query,
~headers=[("Content-Type", "application/json"), ...headers],
baseUrl,
)
|> Lwt.map(
fun
| Ok({Fetch.Response.body, _}) =>
Store.publish(~query, Fetch.Body.toString(body))
| _ => dispatch(Error),
)
|> ignore;
};
let%hook () =
Hooks.effect(
OnMountAndIf(
(prevVariables, nextVariables) => prevVariables != nextVariables,
variables,
),
() => {
let (payload, variables) =
createBody(~query=graphqlQuery, ~variables?);
let unsubscribe = subscribeToStore(payload);
executeRequest(payload, variables);
Some(unsubscribe);
},
);
state;
};
};