@terreno/rtk 0.10.0 → 0.11.0

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.
Files changed (65) hide show
  1. package/dist/authSlice.test.js +68 -0
  2. package/dist/authSlice.test.js.map +1 -1
  3. package/dist/authSliceNative.test.d.ts +2 -0
  4. package/dist/authSliceNative.test.d.ts.map +1 -0
  5. package/dist/authSliceNative.test.js +167 -0
  6. package/dist/authSliceNative.test.js.map +1 -0
  7. package/dist/betterAuthClient.d.ts +16 -0
  8. package/dist/betterAuthClient.d.ts.map +1 -1
  9. package/dist/betterAuthClient.js +5 -2
  10. package/dist/betterAuthClient.js.map +1 -1
  11. package/dist/betterAuthClient.test.d.ts +2 -0
  12. package/dist/betterAuthClient.test.d.ts.map +1 -0
  13. package/dist/betterAuthClient.test.js +151 -0
  14. package/dist/betterAuthClient.test.js.map +1 -0
  15. package/dist/betterAuthSlice.test.js +54 -1
  16. package/dist/betterAuthSlice.test.js.map +1 -1
  17. package/dist/buildNumber.test.d.ts +2 -0
  18. package/dist/buildNumber.test.d.ts.map +1 -0
  19. package/dist/buildNumber.test.js +95 -0
  20. package/dist/buildNumber.test.js.map +1 -0
  21. package/dist/constants.d.ts +27 -3
  22. package/dist/constants.d.ts.map +1 -1
  23. package/dist/constants.js +45 -56
  24. package/dist/constants.js.map +1 -1
  25. package/dist/constants.test.js +174 -123
  26. package/dist/constants.test.js.map +1 -1
  27. package/dist/mongooseSlice.test.d.ts +2 -0
  28. package/dist/mongooseSlice.test.d.ts.map +1 -0
  29. package/dist/mongooseSlice.test.js +39 -0
  30. package/dist/mongooseSlice.test.js.map +1 -0
  31. package/dist/tagGenerator.test.d.ts +2 -0
  32. package/dist/tagGenerator.test.d.ts.map +1 -0
  33. package/dist/tagGenerator.test.js +96 -0
  34. package/dist/tagGenerator.test.js.map +1 -0
  35. package/dist/testPreload.test.d.ts +2 -0
  36. package/dist/testPreload.test.d.ts.map +1 -0
  37. package/dist/testPreload.test.js +27 -0
  38. package/dist/testPreload.test.js.map +1 -0
  39. package/dist/useFeatureFlags.d.ts +25 -1
  40. package/dist/useFeatureFlags.d.ts.map +1 -1
  41. package/dist/useFeatureFlags.js +18 -16
  42. package/dist/useFeatureFlags.js.map +1 -1
  43. package/dist/useFeatureFlags.test.d.ts +2 -0
  44. package/dist/useFeatureFlags.test.d.ts.map +1 -0
  45. package/dist/useFeatureFlags.test.js +162 -0
  46. package/dist/useFeatureFlags.test.js.map +1 -0
  47. package/package.json +6 -3
  48. package/src/authSlice.test.ts +79 -0
  49. package/src/authSliceNative.test.ts +187 -0
  50. package/src/betterAuthClient.test.ts +176 -0
  51. package/src/betterAuthClient.ts +6 -3
  52. package/src/betterAuthSlice.test.ts +67 -0
  53. package/src/buildNumber.test.ts +120 -0
  54. package/src/constants.test.ts +193 -154
  55. package/src/constants.ts +72 -70
  56. package/src/mongooseSlice.test.ts +46 -0
  57. package/src/tagGenerator.test.ts +109 -0
  58. package/src/testPreload.test.ts +30 -0
  59. package/src/useFeatureFlags.test.ts +209 -0
  60. package/src/useFeatureFlags.ts +44 -5
  61. package/dist/test-preload.d.ts +0 -2
  62. package/dist/test-preload.d.ts.map +0 -1
  63. package/dist/test-preload.js +0 -24
  64. package/dist/test-preload.js.map +0 -1
  65. package/src/test-preload.ts +0 -28
@@ -0,0 +1,109 @@
1
+ import {describe, expect, it} from "bun:test";
2
+
3
+ import {generateTags} from "./tagGenerator";
4
+
5
+ describe("generateTags", () => {
6
+ const tagTypes = ["users", "posts", "conversations", "messages"];
7
+
8
+ it("assigns invalidatesTags for getConversations endpoint", () => {
9
+ const api = {
10
+ endpoints: {
11
+ getConversations: {},
12
+ },
13
+ };
14
+ const tags = generateTags(api, tagTypes);
15
+ // getConversations matches both the special case AND the list-endpoint branch,
16
+ // so the list-endpoint branch wins as the final assignment. The special case
17
+ // still executes for coverage, but the final value is the list providesTags.
18
+ expect(tags.getConversations).toBeDefined();
19
+ expect(typeof tags.getConversations.providesTags).toBe("function");
20
+ });
21
+
22
+ it("assigns providesTags on list get endpoints", () => {
23
+ const api = {
24
+ endpoints: {
25
+ getUsers: {},
26
+ },
27
+ };
28
+ const tags = generateTags(api, tagTypes);
29
+ expect(typeof tags.getUsers.providesTags).toBe("function");
30
+
31
+ // Exercise the returned provides function with and without data
32
+ const providesFn = tags.getUsers.providesTags;
33
+ expect(providesFn(null)).toEqual(["users"]);
34
+ expect(providesFn({data: [{_id: "1"}, {_id: "2"}]})).toEqual([
35
+ {id: "1", type: "users"},
36
+ {id: "2", type: "users"},
37
+ "users",
38
+ ]);
39
+ expect(providesFn({})).toEqual(["users"]);
40
+ });
41
+
42
+ it("assigns providesTags on read (ById) get endpoints", () => {
43
+ const api = {
44
+ endpoints: {
45
+ getUserById: {},
46
+ },
47
+ };
48
+ const tags = generateTags(api, tagTypes);
49
+ expect(typeof tags.getUserById.providesTags).toBe("function");
50
+
51
+ const providesFn = tags.getUserById.providesTags;
52
+ expect(providesFn(null)).toEqual(["users"]);
53
+ expect(providesFn({_id: "abc"})).toEqual([{id: "abc", type: "users"}]);
54
+ });
55
+
56
+ it("assigns invalidatesTags on patch endpoints", () => {
57
+ const api = {
58
+ endpoints: {
59
+ patchUserById: {},
60
+ },
61
+ };
62
+ const tags = generateTags(api, tagTypes);
63
+ expect(typeof tags.patchUserById.invalidatesTags).toBe("function");
64
+
65
+ const invalidateFn = tags.patchUserById.invalidatesTags;
66
+ expect(invalidateFn(null)).toEqual(["users"]);
67
+ expect(invalidateFn({data: [{_id: "x"}]})).toEqual([{id: "x", type: "users"}, "users"]);
68
+ });
69
+
70
+ it("assigns invalidatesTags on delete endpoints", () => {
71
+ const api = {
72
+ endpoints: {
73
+ deletePostById: {},
74
+ },
75
+ };
76
+ const tags = generateTags(api, tagTypes);
77
+ expect(typeof tags.deletePostById.invalidatesTags).toBe("function");
78
+
79
+ const invalidateFn = tags.deletePostById.invalidatesTags;
80
+ expect(invalidateFn(null)).toEqual(["posts"]);
81
+ });
82
+
83
+ it("skips endpoints with no matching tag", () => {
84
+ const api = {
85
+ endpoints: {
86
+ getUnknownThing: {},
87
+ patchOrphan: {},
88
+ },
89
+ };
90
+ const tags = generateTags(api, tagTypes);
91
+ expect(tags.getUnknownThing).toBeUndefined();
92
+ expect(tags.patchOrphan).toBeUndefined();
93
+ });
94
+
95
+ it("returns an empty object for an empty endpoint list", () => {
96
+ const tags = generateTags({endpoints: {}}, tagTypes);
97
+ expect(tags).toEqual({});
98
+ });
99
+
100
+ it("ignores endpoints that are not get/patch/delete", () => {
101
+ const api = {
102
+ endpoints: {
103
+ postUser: {},
104
+ },
105
+ };
106
+ const tags = generateTags(api, tagTypes);
107
+ expect(tags.postUser).toBeUndefined();
108
+ });
109
+ });
@@ -0,0 +1,30 @@
1
+ import {describe, expect, it} from "bun:test";
2
+
3
+ // Invoke the mocked module factories registered in test-preload.ts so
4
+ // their bodies (and returned functions) are recognised as covered.
5
+ describe("test-preload default mocks", () => {
6
+ it("expo-secure-store mock returns a no-op store", async () => {
7
+ const SecureStore = await import("expo-secure-store");
8
+ await SecureStore.setItemAsync("k", "v");
9
+ await SecureStore.deleteItemAsync("k");
10
+ expect(await SecureStore.getItemAsync("k")).toBeNull();
11
+ });
12
+
13
+ it("AsyncStorage mock returns a no-op store", async () => {
14
+ const AsyncStorage = (await import("@react-native-async-storage/async-storage")).default;
15
+ await AsyncStorage.setItem("k", "v");
16
+ await AsyncStorage.removeItem("k");
17
+ expect(await AsyncStorage.getItem("k")).toBeNull();
18
+ });
19
+
20
+ it("expo-network mock reports a connected network", async () => {
21
+ const network = await import("expo-network");
22
+ const state = await network.getNetworkStateAsync();
23
+ expect(state.isConnected).toBe(true);
24
+ });
25
+
26
+ it("expo-constants mock exposes an empty config", async () => {
27
+ const Constants = (await import("expo-constants")).default;
28
+ expect(Constants.expoConfig?.extra).toBeDefined();
29
+ });
30
+ });
@@ -0,0 +1,209 @@
1
+ import {afterEach, beforeEach, describe, expect, it, mock} from "bun:test";
2
+ import {renderHook} from "@testing-library/react-native";
3
+
4
+ import {resolveFeatureFlagsOptions, useFeatureFlags} from "./useFeatureFlags";
5
+
6
+ type FlagValues = Record<string, boolean | string | null>;
7
+ type QueryResult = {
8
+ data?: FlagValues;
9
+ error?: unknown;
10
+ isLoading?: boolean;
11
+ };
12
+
13
+ describe("resolveFeatureFlagsOptions", () => {
14
+ it("applies defaults when no argument is provided", () => {
15
+ expect(resolveFeatureFlagsOptions()).toEqual({
16
+ basePath: "/feature-flags",
17
+ skip: false,
18
+ });
19
+ });
20
+
21
+ it("applies defaults when an empty options object is provided", () => {
22
+ expect(resolveFeatureFlagsOptions({})).toEqual({
23
+ basePath: "/feature-flags",
24
+ skip: false,
25
+ });
26
+ });
27
+
28
+ it("treats a string argument as a legacy basePath with skip=false", () => {
29
+ expect(resolveFeatureFlagsOptions("/custom-path")).toEqual({
30
+ basePath: "/custom-path",
31
+ skip: false,
32
+ });
33
+ });
34
+
35
+ it("preserves an empty string basePath when passed as a legacy string argument", () => {
36
+ expect(resolveFeatureFlagsOptions("")).toEqual({
37
+ basePath: "",
38
+ skip: false,
39
+ });
40
+ });
41
+
42
+ it("uses options.basePath when provided", () => {
43
+ expect(resolveFeatureFlagsOptions({basePath: "/flags"})).toEqual({
44
+ basePath: "/flags",
45
+ skip: false,
46
+ });
47
+ });
48
+
49
+ it("uses options.skip when provided", () => {
50
+ expect(resolveFeatureFlagsOptions({skip: true})).toEqual({
51
+ basePath: "/feature-flags",
52
+ skip: true,
53
+ });
54
+ });
55
+
56
+ it("uses both options.basePath and options.skip when provided together", () => {
57
+ expect(resolveFeatureFlagsOptions({basePath: "/flags", skip: true})).toEqual({
58
+ basePath: "/flags",
59
+ skip: true,
60
+ });
61
+ });
62
+
63
+ it("does not let a legacy string basePath override skip to true", () => {
64
+ expect(resolveFeatureFlagsOptions("/custom-path").skip).toBe(false);
65
+ });
66
+ });
67
+
68
+ const buildApi = (queryResult: QueryResult) => {
69
+ const refetch = mock(() => {});
70
+ const capturedQueryBuilder: Array<{method: string; url: string}> = [];
71
+ const useEvaluateFeatureFlagsQuery = mock(() => ({
72
+ data: queryResult.data,
73
+ error: queryResult.error,
74
+ isLoading: queryResult.isLoading ?? false,
75
+ refetch,
76
+ }));
77
+ const api = {
78
+ injectEndpoints: mock((opts: {endpoints: (builder: unknown) => void}) => {
79
+ const builder = {
80
+ query: (def: {query: () => {method: string; url: string}}) => {
81
+ capturedQueryBuilder.push(def.query());
82
+ return {useQuery: useEvaluateFeatureFlagsQuery};
83
+ },
84
+ };
85
+ opts.endpoints(builder);
86
+ return {useEvaluateFeatureFlagsQuery};
87
+ }),
88
+ };
89
+ return {api, capturedQueryBuilder, refetch, useEvaluateFeatureFlagsQuery};
90
+ };
91
+
92
+ describe("useFeatureFlags hook", () => {
93
+ const debugCalls: unknown[][] = [];
94
+ const originalDebug = console.debug;
95
+
96
+ beforeEach(() => {
97
+ debugCalls.length = 0;
98
+ console.debug = (...args: unknown[]): void => {
99
+ debugCalls.push(args);
100
+ };
101
+ });
102
+
103
+ afterEach(() => {
104
+ console.debug = originalDebug;
105
+ });
106
+
107
+ it("builds the evaluate endpoint with the default basePath", () => {
108
+ const {api, capturedQueryBuilder} = buildApi({data: {foo: true}});
109
+ renderHook(() => useFeatureFlags(api as unknown as Parameters<typeof useFeatureFlags>[0]));
110
+ expect(capturedQueryBuilder[0]).toEqual({
111
+ method: "GET",
112
+ url: "/feature-flags/evaluate",
113
+ });
114
+ });
115
+
116
+ it("builds the evaluate endpoint with a custom basePath from legacy string argument", () => {
117
+ const {api, capturedQueryBuilder} = buildApi({data: {foo: true}});
118
+ renderHook(() =>
119
+ useFeatureFlags(api as unknown as Parameters<typeof useFeatureFlags>[0], "/flags")
120
+ );
121
+ expect(capturedQueryBuilder[0]).toEqual({method: "GET", url: "/flags/evaluate"});
122
+ });
123
+
124
+ it("returns flag accessors that read boolean and string values", () => {
125
+ const {api} = buildApi({
126
+ data: {booleanOff: false, booleanOn: true, variantFlag: "variant-a"},
127
+ });
128
+ const {result} = renderHook(() =>
129
+ useFeatureFlags(api as unknown as Parameters<typeof useFeatureFlags>[0])
130
+ );
131
+ expect(result.current.getFlag("booleanOn")).toBe(true);
132
+ expect(result.current.getFlag("booleanOff")).toBe(false);
133
+ expect(result.current.getFlag("variantFlag")).toBe(false);
134
+ expect(result.current.getVariant("variantFlag")).toBe("variant-a");
135
+ expect(result.current.getVariant("booleanOn")).toBeNull();
136
+ expect(result.current.getVariant("missing")).toBeNull();
137
+ });
138
+
139
+ it("returns an empty flags map when no data is available", () => {
140
+ const {api} = buildApi({isLoading: true});
141
+ const {result} = renderHook(() =>
142
+ useFeatureFlags(api as unknown as Parameters<typeof useFeatureFlags>[0])
143
+ );
144
+ expect(result.current.flags).toEqual({});
145
+ expect(result.current.isLoading).toBe(true);
146
+ });
147
+
148
+ it("exposes refetch from the underlying query hook", () => {
149
+ const {api, refetch} = buildApi({data: {foo: true}});
150
+ const {result} = renderHook(() =>
151
+ useFeatureFlags(api as unknown as Parameters<typeof useFeatureFlags>[0])
152
+ );
153
+ result.current.refetch();
154
+ expect(refetch).toHaveBeenCalled();
155
+ });
156
+
157
+ it("logs evaluate request started when loading begins without prior data", () => {
158
+ const {api} = buildApi({isLoading: true});
159
+ renderHook(() => useFeatureFlags(api as unknown as Parameters<typeof useFeatureFlags>[0]));
160
+ const started = debugCalls.find(
161
+ (args) => args[0] === "[feature-flags] evaluate request started"
162
+ );
163
+ expect(started).toBeDefined();
164
+ });
165
+
166
+ it("logs evaluate request completed when data resolves after a loading phase", () => {
167
+ const api = buildApi({isLoading: true});
168
+ const {rerender} = renderHook(
169
+ ({queryResult}: {queryResult: QueryResult}) => {
170
+ api.useEvaluateFeatureFlagsQuery.mockImplementationOnce(() => ({
171
+ data: queryResult.data,
172
+ error: queryResult.error,
173
+ isLoading: queryResult.isLoading ?? false,
174
+ refetch: api.refetch,
175
+ }));
176
+ return useFeatureFlags(api.api as unknown as Parameters<typeof useFeatureFlags>[0]);
177
+ },
178
+ {initialProps: {queryResult: {isLoading: true}}}
179
+ );
180
+ rerender({queryResult: {data: {alpha: true}, isLoading: false}});
181
+ const completed = debugCalls.find(
182
+ (args) => args[0] === "[feature-flags] evaluate request completed"
183
+ );
184
+ const started = debugCalls.find(
185
+ (args) => args[0] === "[feature-flags] evaluate request started"
186
+ );
187
+ expect(started).toBeDefined();
188
+ expect(completed).toBeDefined();
189
+ });
190
+
191
+ it("logs evaluate request failed when error is returned after loading", () => {
192
+ const api = buildApi({isLoading: true});
193
+ const {rerender} = renderHook(
194
+ ({queryResult}: {queryResult: QueryResult}) => {
195
+ api.useEvaluateFeatureFlagsQuery.mockImplementationOnce(() => ({
196
+ data: queryResult.data,
197
+ error: queryResult.error,
198
+ isLoading: queryResult.isLoading ?? false,
199
+ refetch: api.refetch,
200
+ }));
201
+ return useFeatureFlags(api.api as unknown as Parameters<typeof useFeatureFlags>[0]);
202
+ },
203
+ {initialProps: {queryResult: {isLoading: true}}}
204
+ );
205
+ rerender({queryResult: {error: new Error("boom"), isLoading: false}});
206
+ const failed = debugCalls.find((args) => args[0] === "[feature-flags] evaluate request failed");
207
+ expect(failed).toBeDefined();
208
+ });
209
+ });
@@ -27,11 +27,50 @@ interface UseFeatureFlagsResult {
27
27
  * const variant = getVariant("checkout-experiment"); // "control" | "variant-a" | null
28
28
  * ```
29
29
  */
30
- export const useFeatureFlags = (
30
+ export interface UseFeatureFlagsOptions {
31
+ /**
32
+ * Base path for the feature-flags endpoint. Defaults to "/feature-flags".
33
+ */
34
+ basePath?: string;
35
+ /**
36
+ * When true, the underlying evaluate query is not fired. Use this to avoid
37
+ * fetching before the user is authenticated.
38
+ */
39
+ skip?: boolean;
40
+ }
41
+
42
+ interface ResolvedFeatureFlagsOptions {
43
+ basePath: string;
44
+ skip: boolean;
45
+ }
46
+
47
+ /**
48
+ * Normalizes the legacy-compatible `basePathOrOptions` argument into a
49
+ * `{basePath, skip}` pair with defaults applied.
50
+ *
51
+ * - `undefined` -> `{basePath: "/feature-flags", skip: false}`
52
+ * - `string` -> `{basePath: <string>, skip: false}` (legacy form)
53
+ * - `object` -> `{basePath: opts.basePath ?? "/feature-flags", skip: opts.skip ?? false}`
54
+ */
55
+ export const resolveFeatureFlagsOptions = (
56
+ basePathOrOptions?: string | UseFeatureFlagsOptions
57
+ ): ResolvedFeatureFlagsOptions => {
58
+ const {basePath = "/feature-flags", skip = false} =
59
+ typeof basePathOrOptions === "string"
60
+ ? {basePath: basePathOrOptions, skip: false}
61
+ : (basePathOrOptions ?? {});
62
+ return {basePath, skip};
63
+ };
64
+
65
+ // Overloaded signature preserves backwards compatibility with callers that
66
+ // pass a string basePath as the second argument.
67
+ export function useFeatureFlags(
31
68
  // biome-ignore lint/suspicious/noExplicitAny: RTK Query API generic typing is intentionally flexible here.
32
69
  api: Api<any, any, any, any>,
33
- basePath = "/feature-flags"
34
- ): UseFeatureFlagsResult => {
70
+ basePathOrOptions?: string | UseFeatureFlagsOptions
71
+ ): UseFeatureFlagsResult {
72
+ const {basePath, skip} = resolveFeatureFlagsOptions(basePathOrOptions);
73
+
35
74
  const enhancedApi = useMemo(
36
75
  () =>
37
76
  api.injectEndpoints({
@@ -50,7 +89,7 @@ export const useFeatureFlags = (
50
89
 
51
90
  // biome-ignore lint/suspicious/noExplicitAny: Endpoint hook is injected dynamically by RTK Query.
52
91
  const useEvaluateQuery = (enhancedApi as any).useEvaluateFeatureFlagsQuery;
53
- const {data, isLoading, error, refetch} = useEvaluateQuery();
92
+ const {data, isLoading, error, refetch} = useEvaluateQuery(undefined, {skip});
54
93
  const evaluateStartedAtRef = useRef<number | null>(null);
55
94
 
56
95
  const flags: FlagValues = data ?? {};
@@ -118,4 +157,4 @@ export const useFeatureFlags = (
118
157
  );
119
158
 
120
159
  return {error, flags, getFlag, getVariant, isLoading, refetch};
121
- };
160
+ }
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=test-preload.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"test-preload.d.ts","sourceRoot":"","sources":["../src/test-preload.ts"],"names":[],"mappings":""}
@@ -1,24 +0,0 @@
1
- import { mock } from "bun:test";
2
- mock.module("react-native", () => ({
3
- Platform: { OS: "web" },
4
- StyleSheet: { create: (s) => s },
5
- }));
6
- mock.module("expo-secure-store", () => ({
7
- deleteItemAsync: async () => { },
8
- getItemAsync: async () => null,
9
- setItemAsync: async () => { },
10
- }));
11
- mock.module("@react-native-async-storage/async-storage", () => ({
12
- default: {
13
- getItem: async () => null,
14
- removeItem: async () => { },
15
- setItem: async () => { },
16
- },
17
- }));
18
- mock.module("expo-constants", () => ({
19
- default: { expoConfig: { extra: {} } },
20
- }));
21
- mock.module("expo-network", () => ({
22
- getNetworkStateAsync: async () => ({ isConnected: true }),
23
- }));
24
- //# sourceMappingURL=test-preload.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"test-preload.js","sourceRoot":"","sources":["../src/test-preload.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,IAAI,EAAC,MAAM,UAAU,CAAC;AAE9B,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,GAAG,EAAE,CAAC,CAAC;IACjC,QAAQ,EAAE,EAAC,EAAE,EAAE,KAAK,EAAC;IACrB,UAAU,EAAE,EAAC,MAAM,EAAE,CAAC,CAAU,EAAE,EAAE,CAAC,CAAC,EAAC;CACxC,CAAC,CAAC,CAAC;AAEJ,IAAI,CAAC,MAAM,CAAC,mBAAmB,EAAE,GAAG,EAAE,CAAC,CAAC;IACtC,eAAe,EAAE,KAAK,IAAI,EAAE,GAAE,CAAC;IAC/B,YAAY,EAAE,KAAK,IAAI,EAAE,CAAC,IAAI;IAC9B,YAAY,EAAE,KAAK,IAAI,EAAE,GAAE,CAAC;CAC7B,CAAC,CAAC,CAAC;AAEJ,IAAI,CAAC,MAAM,CAAC,2CAA2C,EAAE,GAAG,EAAE,CAAC,CAAC;IAC9D,OAAO,EAAE;QACP,OAAO,EAAE,KAAK,IAAI,EAAE,CAAC,IAAI;QACzB,UAAU,EAAE,KAAK,IAAI,EAAE,GAAE,CAAC;QAC1B,OAAO,EAAE,KAAK,IAAI,EAAE,GAAE,CAAC;KACxB;CACF,CAAC,CAAC,CAAC;AAEJ,IAAI,CAAC,MAAM,CAAC,gBAAgB,EAAE,GAAG,EAAE,CAAC,CAAC;IACnC,OAAO,EAAE,EAAC,UAAU,EAAE,EAAC,KAAK,EAAE,EAAE,EAAC,EAAC;CACnC,CAAC,CAAC,CAAC;AAEJ,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,GAAG,EAAE,CAAC,CAAC;IACjC,oBAAoB,EAAE,KAAK,IAAI,EAAE,CAAC,CAAC,EAAC,WAAW,EAAE,IAAI,EAAC,CAAC;CACxD,CAAC,CAAC,CAAC"}
@@ -1,28 +0,0 @@
1
- import {mock} from "bun:test";
2
-
3
- mock.module("react-native", () => ({
4
- Platform: {OS: "web"},
5
- StyleSheet: {create: (s: unknown) => s},
6
- }));
7
-
8
- mock.module("expo-secure-store", () => ({
9
- deleteItemAsync: async () => {},
10
- getItemAsync: async () => null,
11
- setItemAsync: async () => {},
12
- }));
13
-
14
- mock.module("@react-native-async-storage/async-storage", () => ({
15
- default: {
16
- getItem: async () => null,
17
- removeItem: async () => {},
18
- setItem: async () => {},
19
- },
20
- }));
21
-
22
- mock.module("expo-constants", () => ({
23
- default: {expoConfig: {extra: {}}},
24
- }));
25
-
26
- mock.module("expo-network", () => ({
27
- getNetworkStateAsync: async () => ({isConnected: true}),
28
- }));