@shwfed/nuxt 0.11.10 → 0.11.12

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.
package/dist/module.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@shwfed/nuxt",
3
3
  "configKey": "shwfed",
4
- "version": "0.11.10",
4
+ "version": "0.11.12",
5
5
  "builder": {
6
6
  "@nuxt/module-builder": "1.0.2",
7
7
  "unbuild": "3.6.1"
@@ -1,6 +1,6 @@
1
1
  import { Context, Deferred, Effect, Option, type Scope } from 'effect';
2
2
  import { type VNode } from 'vue';
3
- export type OverlayResult = Option.Option<unknown>;
3
+ export type OverlayResult<T = unknown> = Option.Option<T>;
4
4
  export type OverlayShellProps = Readonly<{
5
5
  title?: string;
6
6
  description?: string;
@@ -42,11 +42,11 @@ type OverlaySession = {
42
42
  props: Record<string, unknown>;
43
43
  deferred: Deferred.Deferred<OverlayResult, never>;
44
44
  };
45
- export type OverlayHandle = Readonly<{
45
+ export type OverlayHandle<T = unknown> = Readonly<{
46
46
  sessionId: string;
47
47
  patch: (patch: OverlaySessionPatch) => Effect.Effect<void>;
48
48
  close: (...args: [value?: unknown]) => Effect.Effect<void>;
49
- result: Effect.Effect<OverlayResult>;
49
+ result: Effect.Effect<OverlayResult<T>>;
50
50
  }>;
51
51
  export type OverlayDefinitionNotFoundError = Readonly<{
52
52
  _tag: 'OverlayDefinitionNotFoundError';
@@ -64,7 +64,7 @@ export interface OverlayRuntime {
64
64
  closeSync: (sessionId: string, ...args: [value?: unknown]) => void;
65
65
  closeTopSync: () => void;
66
66
  closeAllSync: () => void;
67
- open: (definitionId: string, options?: OverlaySessionInput) => Effect.Effect<OverlayHandle, OverlayDefinitionNotFoundError, Scope.Scope>;
67
+ open: <T>(definitionId: string, options?: OverlaySessionInput) => Effect.Effect<OverlayHandle<T>, OverlayDefinitionNotFoundError, Scope.Scope>;
68
68
  patch: (sessionId: string, patch: OverlaySessionPatch) => Effect.Effect<void>;
69
69
  close: (sessionId: string, ...args: [value?: unknown]) => Effect.Effect<void>;
70
70
  closeTop: () => Effect.Effect<void>;
@@ -75,7 +75,7 @@ export declare class OverlayService extends OverlayService_base {
75
75
  }
76
76
  export declare function useOverlay(): OverlayRuntime;
77
77
  export declare function provideOverlay<A, E, R>(effect: Effect.Effect<A, E, R>, runtime?: OverlayRuntime): Effect.Effect<A, E, Exclude<R, OverlayService>>;
78
- export declare function openOverlay(definitionId: string, options?: OverlaySessionInput): Effect.Effect<OverlayHandle, OverlayDefinitionNotFoundError, OverlayService | Scope.Scope>;
78
+ export declare function openOverlay<T>(definitionId: string, options?: OverlaySessionInput): Effect.Effect<OverlayHandle<T>, OverlayDefinitionNotFoundError, OverlayService | Scope.Scope>;
79
79
  export declare function patchOverlay(sessionId: string, patch: OverlaySessionPatch): Effect.Effect<void, never, OverlayService>;
80
80
  export declare function closeOverlay(sessionId: string, ...args: [value?: unknown]): Effect.Effect<void, never, OverlayService>;
81
81
  export declare function closeTopOverlay(): Effect.Effect<void, never, OverlayService>;
@@ -136,25 +136,8 @@ function createOverlayRuntime() {
136
136
  result: Deferred.await(session.deferred)
137
137
  };
138
138
  }
139
- const runtime = {
140
- get definitions() {
141
- return definitions;
142
- },
143
- get sessions() {
144
- return sessions.value;
145
- },
146
- get topSession() {
147
- return sessions.value.at(-1);
148
- },
149
- registerDefinition,
150
- unregisterDefinition,
151
- syncDefinitions,
152
- isOpen: (sessionId) => findSessionIndex(sessionId) >= 0,
153
- patchSync,
154
- closeSync,
155
- closeTopSync,
156
- closeAllSync,
157
- open: (definitionId, options) => Effect.acquireRelease(
139
+ function open(definitionId, options) {
140
+ return Effect.acquireRelease(
158
141
  Effect.gen(function* () {
159
142
  const definition = getDefinition(definitionId);
160
143
  if (!definition) {
@@ -194,7 +177,27 @@ function createOverlayRuntime() {
194
177
  return createHandle(session);
195
178
  }),
196
179
  (handle) => closeInternal(handle.sessionId, Option.none())
197
- ),
180
+ );
181
+ }
182
+ const runtime = {
183
+ get definitions() {
184
+ return definitions;
185
+ },
186
+ get sessions() {
187
+ return sessions.value;
188
+ },
189
+ get topSession() {
190
+ return sessions.value.at(-1);
191
+ },
192
+ registerDefinition,
193
+ unregisterDefinition,
194
+ syncDefinitions,
195
+ isOpen: (sessionId) => findSessionIndex(sessionId) >= 0,
196
+ patchSync,
197
+ closeSync,
198
+ closeTopSync,
199
+ closeAllSync,
200
+ open,
198
201
  patch: (sessionId, patch) => Effect.sync(() => {
199
202
  patchSync(sessionId, patch);
200
203
  }),
@@ -1,6 +1,17 @@
1
+ import { Effect } from 'effect';
2
+ type JsonQueryValue = string | number | boolean | null | undefined;
3
+ type JsonOptions = Readonly<{
4
+ method?: string;
5
+ headers?: HeadersInit;
6
+ query?: Readonly<Record<string, JsonQueryValue | ReadonlyArray<JsonQueryValue>>>;
7
+ body?: FormData | unknown;
8
+ signal?: AbortSignal;
9
+ }>;
1
10
  declare const _default: import("#app").Plugin<{
2
11
  api: import("nitropack").$Fetch<unknown, import("nitropack").NitroFetchRequest>;
12
+ json: <T>(input: string, options?: JsonOptions) => Effect.Effect<T, Error>;
3
13
  }> & import("#app").ObjectPlugin<{
4
14
  api: import("nitropack").$Fetch<unknown, import("nitropack").NitroFetchRequest>;
15
+ json: <T>(input: string, options?: JsonOptions) => Effect.Effect<T, Error>;
5
16
  }>;
6
17
  export default _default;
@@ -1,5 +1,6 @@
1
1
  import { defineNuxtPlugin, useNuxtApp, useRuntimeConfig } from "#app";
2
2
  import { useNavigatorLanguage } from "@vueuse/core";
3
+ import { Effect } from "effect";
3
4
  import z from "zod";
4
5
  let isRedirectingAfterTokenExpiry = false;
5
6
  const processedResponses = /* @__PURE__ */ new WeakSet();
@@ -22,7 +23,7 @@ export default defineNuxtPlugin({
22
23
  const isExpired = z.boolean().parse($dsl.evaluate`${config.config.api.expired}`({
23
24
  status: BigInt(response.status),
24
25
  ok: response.ok,
25
- data: response._data ?? null
26
+ data: response.data ?? response._data ?? null
26
27
  }));
27
28
  if (!isExpired || isRedirectingAfterTokenExpiry) {
28
29
  return;
@@ -38,27 +39,82 @@ export default defineNuxtPlugin({
38
39
  console.error(e);
39
40
  }
40
41
  }
42
+ function applyRequestHeaders(headers) {
43
+ if (locale.isSupported && locale.language.value) {
44
+ headers.set("Accept-Language", locale.language.value);
45
+ }
46
+ if (typeof config.config?.api?.header === "string") {
47
+ try {
48
+ for (const [key, value] of Object.entries(
49
+ z.record(z.string(), z.string()).parse($dsl.evaluate`${config.config.api.header}`())
50
+ )) {
51
+ headers.set(key, value);
52
+ }
53
+ } catch (e) {
54
+ console.error(e);
55
+ }
56
+ }
57
+ const name = config.headers.token || "Authorization";
58
+ headers.set(name, sessionStorage.getItem("token") ?? "");
59
+ }
60
+ function createRequestUrl(input, query) {
61
+ const url = new URL(input, config.api.host || window.location.href);
62
+ if (!query) {
63
+ return url.toString();
64
+ }
65
+ for (const [key, value] of Object.entries(query)) {
66
+ if (value === void 0) {
67
+ continue;
68
+ }
69
+ if (Array.isArray(value)) {
70
+ for (const entry of value) {
71
+ if (entry !== void 0) {
72
+ url.searchParams.append(key, String(entry));
73
+ }
74
+ }
75
+ continue;
76
+ }
77
+ url.searchParams.append(key, String(value));
78
+ }
79
+ return url.toString();
80
+ }
81
+ function createRequestBody(body, headers) {
82
+ if (body === void 0) {
83
+ return void 0;
84
+ }
85
+ if (body instanceof FormData) {
86
+ return body;
87
+ }
88
+ if (!headers.has("Content-Type")) {
89
+ headers.set("Content-Type", "application/json");
90
+ }
91
+ return JSON.stringify(body);
92
+ }
93
+ function toError(error) {
94
+ if (error instanceof Error) {
95
+ return error;
96
+ }
97
+ return new Error(String(error));
98
+ }
99
+ async function handleExpiredFetchResponse(response) {
100
+ let data = null;
101
+ try {
102
+ data = await response.clone().json();
103
+ } catch {
104
+ data = null;
105
+ }
106
+ handleExpiredResponse({
107
+ status: response.status,
108
+ ok: response.ok,
109
+ data
110
+ });
111
+ }
41
112
  const api = $fetch.create({
42
113
  baseURL: config.api.host,
43
114
  onRequest: ({
44
115
  options
45
116
  }) => {
46
- if (locale.isSupported && locale.language.value) {
47
- options.headers.set("Accept-Language", locale.language.value);
48
- }
49
- if (typeof config.config?.api?.header === "string") {
50
- try {
51
- for (const [key, value] of Object.entries(
52
- z.record(z.string(), z.string()).parse($dsl.evaluate`${config.config.api.header}`())
53
- )) {
54
- options.headers.set(key, value);
55
- }
56
- } catch (e) {
57
- console.error(e);
58
- }
59
- }
60
- const name = config.headers.token || "Authorization";
61
- options.headers.set(name, sessionStorage.getItem("token") ?? "");
117
+ applyRequestHeaders(options.headers);
62
118
  },
63
119
  onResponse: ({ response }) => {
64
120
  handleExpiredResponse(response);
@@ -67,9 +123,34 @@ export default defineNuxtPlugin({
67
123
  handleExpiredResponse(response);
68
124
  }
69
125
  });
126
+ function json(input, options = {}) {
127
+ const headers = new Headers(options.headers);
128
+ applyRequestHeaders(headers);
129
+ return Effect.tryPromise({
130
+ try: (signal) => fetch(createRequestUrl(input, options.query), {
131
+ method: options.method,
132
+ headers,
133
+ body: createRequestBody(options.body, headers),
134
+ signal: options.signal ?? signal
135
+ }),
136
+ catch: toError
137
+ }).pipe(
138
+ Effect.flatMap((response) => Effect.tryPromise({
139
+ try: async () => {
140
+ await handleExpiredFetchResponse(response);
141
+ if (!response.ok) {
142
+ throw new Error(`Request failed with status ${response.status}`);
143
+ }
144
+ return await response.json();
145
+ },
146
+ catch: toError
147
+ }))
148
+ );
149
+ }
70
150
  return {
71
151
  provide: {
72
- api
152
+ api,
153
+ json
73
154
  }
74
155
  };
75
156
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@shwfed/nuxt",
3
- "version": "0.11.10",
3
+ "version": "0.11.12",
4
4
  "description": "",
5
5
  "license": "MIT",
6
6
  "type": "module",