@trpc/client 11.2.0 → 11.2.1-canary.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,15 @@
1
+ import { type AnyRouter, type ErrorHandlerOptions, type inferClientTypes, type inferRouterContext } from '@trpc/server/unstable-core-do-not-import';
2
+ import type { TransformerOptions } from '../internals/transformer';
3
+ import type { TRPCLink } from './types';
4
+ export type LocalLinkOptions<TRouter extends AnyRouter> = {
5
+ router: TRouter;
6
+ createContext: () => Promise<inferRouterContext<TRouter>>;
7
+ onError?: (opts: ErrorHandlerOptions<inferRouterContext<TRouter>>) => void;
8
+ } & TransformerOptions<inferClientTypes<TRouter>>;
9
+ /**
10
+ * localLink is a terminating link that allows you to make tRPC procedure calls directly in your application without going through HTTP.
11
+ *
12
+ * @see https://trpc.io/docs/links/localLink
13
+ */
14
+ export declare function experimental_localLink<TRouter extends AnyRouter>(opts: LocalLinkOptions<TRouter>): TRPCLink<TRouter>;
15
+ //# sourceMappingURL=localLink.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"localLink.d.ts","sourceRoot":"","sources":["../../src/links/localLink.ts"],"names":[],"mappings":"AAOA,OAAO,EAQL,KAAK,SAAS,EACd,KAAK,mBAAmB,EACxB,KAAK,gBAAgB,EACrB,KAAK,kBAAkB,EACxB,MAAM,0CAA0C,CAAC;AAIlD,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,0BAA0B,CAAC;AAGnE,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAExC,MAAM,MAAM,gBAAgB,CAAC,OAAO,SAAS,SAAS,IAAI;IACxD,MAAM,EAAE,OAAO,CAAC;IAChB,aAAa,EAAE,MAAM,OAAO,CAAC,kBAAkB,CAAC,OAAO,CAAC,CAAC,CAAC;IAC1D,OAAO,CAAC,EAAE,CAAC,IAAI,EAAE,mBAAmB,CAAC,kBAAkB,CAAC,OAAO,CAAC,CAAC,KAAK,IAAI,CAAC;CAC5E,GAAG,kBAAkB,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC,CAAC;AAElD;;;;GAIG;AACH,wBAAgB,sBAAsB,CAAC,OAAO,SAAS,SAAS,EAC9D,IAAI,EAAE,gBAAgB,CAAC,OAAO,CAAC,GAC9B,QAAQ,CAAC,OAAO,CAAC,CA2OnB"}
@@ -0,0 +1,333 @@
1
+ 'use strict';
2
+
3
+ var server = require('@trpc/server');
4
+ var observable = require('@trpc/server/observable');
5
+ var rpc = require('@trpc/server/rpc');
6
+ var unstableCoreDoNotImport = require('@trpc/server/unstable-core-do-not-import');
7
+ var inputWithTrackedEventId = require('../internals/inputWithTrackedEventId.js');
8
+ var signals = require('../internals/signals.js');
9
+ var transformer = require('../internals/transformer.js');
10
+ var TRPCClientError = require('../TRPCClientError.js');
11
+
12
+ function _ts_add_disposable_resource(env, value, async) {
13
+ if (value !== null && value !== void 0) {
14
+ if (typeof value !== "object" && typeof value !== "function") throw new TypeError("Object expected.");
15
+ var dispose, inner;
16
+ if (async) {
17
+ if (!Symbol.asyncDispose) throw new TypeError("Symbol.asyncDispose is not defined.");
18
+ dispose = value[Symbol.asyncDispose];
19
+ }
20
+ if (dispose === void 0) {
21
+ if (!Symbol.dispose) throw new TypeError("Symbol.dispose is not defined.");
22
+ dispose = value[Symbol.dispose];
23
+ if (async) inner = dispose;
24
+ }
25
+ if (typeof dispose !== "function") throw new TypeError("Object not disposable.");
26
+ if (inner) dispose = function() {
27
+ try {
28
+ inner.call(this);
29
+ } catch (e) {
30
+ return Promise.reject(e);
31
+ }
32
+ };
33
+ env.stack.push({
34
+ value: value,
35
+ dispose: dispose,
36
+ async: async
37
+ });
38
+ } else if (async) {
39
+ env.stack.push({
40
+ async: true
41
+ });
42
+ }
43
+ return value;
44
+ }
45
+ function _ts_dispose_resources(env) {
46
+ var _SuppressedError = typeof SuppressedError === "function" ? SuppressedError : function(error, suppressed, message) {
47
+ var e = new Error(message);
48
+ return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e;
49
+ };
50
+ return (_ts_dispose_resources = function _ts_dispose_resources(env) {
51
+ function fail(e) {
52
+ env.error = env.hasError ? new _SuppressedError(e, env.error, "An error was suppressed during disposal.") : e;
53
+ env.hasError = true;
54
+ }
55
+ var r, s = 0;
56
+ function next() {
57
+ while(r = env.stack.pop()){
58
+ try {
59
+ if (!r.async && s === 1) return s = 0, env.stack.push(r), Promise.resolve().then(next);
60
+ if (r.dispose) {
61
+ var result = r.dispose.call(r.value);
62
+ if (r.async) return s |= 2, Promise.resolve(result).then(next, function(e) {
63
+ fail(e);
64
+ return next();
65
+ });
66
+ } else s |= 1;
67
+ } catch (e) {
68
+ fail(e);
69
+ }
70
+ }
71
+ if (s === 1) return env.hasError ? Promise.reject(env.error) : Promise.resolve();
72
+ if (env.hasError) throw env.error;
73
+ }
74
+ return next();
75
+ })(env);
76
+ }
77
+ /**
78
+ * localLink is a terminating link that allows you to make tRPC procedure calls directly in your application without going through HTTP.
79
+ *
80
+ * @see https://trpc.io/docs/links/localLink
81
+ */ function experimental_localLink(opts) {
82
+ const transformer$1 = transformer.getTransformer(opts.transformer);
83
+ const transformChunk = (chunk)=>{
84
+ if (opts.transformer) {
85
+ // assume transformer will do the right thing
86
+ return chunk;
87
+ }
88
+ // Special case for undefined, because `JSON.stringify(undefined)` throws
89
+ if (chunk === undefined) {
90
+ return chunk;
91
+ }
92
+ const serialized = JSON.stringify(transformer$1.input.serialize(chunk));
93
+ const deserialized = JSON.parse(transformer$1.output.deserialize(serialized));
94
+ return deserialized;
95
+ };
96
+ return ()=>({ op })=>observable.observable((observer)=>{
97
+ let ctx = undefined;
98
+ const ac = new AbortController();
99
+ const signal = signals.raceAbortSignals(op.signal, ac.signal);
100
+ const signalPromise = signals.abortSignalToPromise(signal);
101
+ signalPromise.catch(()=>{
102
+ // prevent unhandled rejection
103
+ });
104
+ let input = op.input;
105
+ async function runProcedure(newInput) {
106
+ input = newInput;
107
+ ctx = await opts.createContext();
108
+ return unstableCoreDoNotImport.callProcedure({
109
+ router: opts.router,
110
+ path: op.path,
111
+ getRawInput: async ()=>newInput,
112
+ ctx,
113
+ type: op.type,
114
+ signal
115
+ });
116
+ }
117
+ function onErrorCallback(cause) {
118
+ if (unstableCoreDoNotImport.isAbortError(cause)) {
119
+ return;
120
+ }
121
+ opts.onError?.({
122
+ error: server.getTRPCErrorFromUnknown(cause),
123
+ type: op.type,
124
+ path: op.path,
125
+ input,
126
+ ctx
127
+ });
128
+ }
129
+ function coerceToTRPCClientError(cause) {
130
+ if (TRPCClientError.isTRPCClientError(cause)) {
131
+ return cause;
132
+ }
133
+ const error = server.getTRPCErrorFromUnknown(cause);
134
+ const shape = server.getTRPCErrorShape({
135
+ config: opts.router._def._config,
136
+ ctx,
137
+ error,
138
+ input,
139
+ path: op.path,
140
+ type: op.type
141
+ });
142
+ return TRPCClientError.TRPCClientError.from({
143
+ error: transformChunk(shape)
144
+ });
145
+ }
146
+ unstableCoreDoNotImport.run(async ()=>{
147
+ switch(op.type){
148
+ case 'query':
149
+ case 'mutation':
150
+ {
151
+ const result = await runProcedure(op.input);
152
+ if (!unstableCoreDoNotImport.isAsyncIterable(result)) {
153
+ observer.next({
154
+ result: {
155
+ data: transformChunk(result)
156
+ }
157
+ });
158
+ observer.complete();
159
+ break;
160
+ }
161
+ observer.next({
162
+ result: {
163
+ data: async function*() {
164
+ const env = {
165
+ stack: [],
166
+ error: void 0,
167
+ hasError: false
168
+ };
169
+ try {
170
+ const iterator = _ts_add_disposable_resource(env, unstableCoreDoNotImport.iteratorResource(result), true);
171
+ ;
172
+ const _finally = _ts_add_disposable_resource(env, unstableCoreDoNotImport.makeResource({}, ()=>{
173
+ observer.complete();
174
+ }), false);
175
+ ;
176
+ try {
177
+ while(true){
178
+ const res = await Promise.race([
179
+ iterator.next(),
180
+ signalPromise
181
+ ]);
182
+ if (res.done) {
183
+ return transformChunk(res.value);
184
+ }
185
+ yield transformChunk(res.value);
186
+ }
187
+ } catch (cause) {
188
+ onErrorCallback(cause);
189
+ throw coerceToTRPCClientError(cause);
190
+ }
191
+ } catch (e) {
192
+ env.error = e;
193
+ env.hasError = true;
194
+ } finally{
195
+ const result = _ts_dispose_resources(env);
196
+ if (result) await result;
197
+ }
198
+ }()
199
+ }
200
+ });
201
+ break;
202
+ }
203
+ case 'subscription':
204
+ {
205
+ const env = {
206
+ stack: [],
207
+ error: void 0,
208
+ hasError: false
209
+ };
210
+ try {
211
+ const connectionState = observable.behaviorSubject({
212
+ type: 'state',
213
+ state: 'connecting',
214
+ error: null
215
+ });
216
+ const connectionSub = connectionState.subscribe({
217
+ next (state) {
218
+ observer.next({
219
+ result: state
220
+ });
221
+ }
222
+ });
223
+ let lastEventId = undefined;
224
+ const _finally = _ts_add_disposable_resource(env, unstableCoreDoNotImport.makeResource({}, async ()=>{
225
+ observer.complete();
226
+ connectionState.next({
227
+ type: 'state',
228
+ state: 'idle',
229
+ error: null
230
+ });
231
+ connectionSub.unsubscribe();
232
+ }), false);
233
+ ;
234
+ while(true){
235
+ const env = {
236
+ stack: [],
237
+ error: void 0,
238
+ hasError: false
239
+ };
240
+ try {
241
+ const result = await runProcedure(inputWithTrackedEventId.inputWithTrackedEventId(op.input, lastEventId));
242
+ if (!unstableCoreDoNotImport.isAsyncIterable(result)) {
243
+ throw new Error('Expected an async iterable');
244
+ }
245
+ const iterator = _ts_add_disposable_resource(env, unstableCoreDoNotImport.iteratorResource(result), true);
246
+ ;
247
+ observer.next({
248
+ result: {
249
+ type: 'started'
250
+ }
251
+ });
252
+ connectionState.next({
253
+ type: 'state',
254
+ state: 'pending',
255
+ error: null
256
+ });
257
+ // Use a while loop to handle errors and reconnects
258
+ while(true){
259
+ let res;
260
+ try {
261
+ res = await Promise.race([
262
+ iterator.next(),
263
+ signalPromise
264
+ ]);
265
+ } catch (cause) {
266
+ if (unstableCoreDoNotImport.isAbortError(cause)) {
267
+ return;
268
+ }
269
+ const error = server.getTRPCErrorFromUnknown(cause);
270
+ if (!unstableCoreDoNotImport.retryableRpcCodes.includes(rpc.TRPC_ERROR_CODES_BY_KEY[error.code])) {
271
+ throw coerceToTRPCClientError(error);
272
+ }
273
+ onErrorCallback(error);
274
+ connectionState.next({
275
+ type: 'state',
276
+ state: 'connecting',
277
+ error: coerceToTRPCClientError(error)
278
+ });
279
+ break;
280
+ }
281
+ if (res.done) {
282
+ return;
283
+ }
284
+ let chunk;
285
+ if (server.isTrackedEnvelope(res.value)) {
286
+ lastEventId = res.value[0];
287
+ chunk = {
288
+ id: res.value[0],
289
+ data: {
290
+ id: res.value[0],
291
+ data: res.value[1]
292
+ }
293
+ };
294
+ } else {
295
+ chunk = {
296
+ data: res.value
297
+ };
298
+ }
299
+ observer.next({
300
+ result: {
301
+ ...chunk,
302
+ data: transformChunk(chunk.data)
303
+ }
304
+ });
305
+ }
306
+ } catch (e) {
307
+ env.error = e;
308
+ env.hasError = true;
309
+ } finally{
310
+ const result = _ts_dispose_resources(env);
311
+ if (result) await result;
312
+ }
313
+ }
314
+ break;
315
+ } catch (e) {
316
+ env.error = e;
317
+ env.hasError = true;
318
+ } finally{
319
+ _ts_dispose_resources(env);
320
+ }
321
+ }
322
+ }
323
+ }).catch((cause)=>{
324
+ onErrorCallback(cause);
325
+ observer.error(coerceToTRPCClientError(cause));
326
+ });
327
+ return ()=>{
328
+ ac.abort();
329
+ };
330
+ });
331
+ }
332
+
333
+ exports.experimental_localLink = experimental_localLink;