@effect-app/vue 4.0.0-beta.16 → 4.0.0-beta.161
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/CHANGELOG.md +1023 -0
- package/dist/commander.d.ts +370 -0
- package/dist/commander.d.ts.map +1 -0
- package/dist/commander.js +591 -0
- package/dist/confirm.d.ts +19 -0
- package/dist/confirm.d.ts.map +1 -0
- package/dist/confirm.js +24 -0
- package/dist/errorReporter.d.ts +4 -4
- package/dist/errorReporter.d.ts.map +1 -1
- package/dist/errorReporter.js +12 -18
- package/dist/form.d.ts +13 -4
- package/dist/form.d.ts.map +1 -1
- package/dist/form.js +41 -12
- package/dist/index.d.ts +1 -1
- package/dist/intl.d.ts +15 -0
- package/dist/intl.d.ts.map +1 -0
- package/dist/intl.js +9 -0
- package/dist/lib.d.ts +2 -1
- package/dist/lib.d.ts.map +1 -1
- package/dist/lib.js +32 -1
- package/dist/makeClient.d.ts +97 -276
- package/dist/makeClient.d.ts.map +1 -1
- package/dist/makeClient.js +94 -353
- package/dist/makeContext.d.ts +1 -1
- package/dist/makeContext.d.ts.map +1 -1
- package/dist/makeIntl.d.ts +1 -1
- package/dist/makeIntl.d.ts.map +1 -1
- package/dist/makeUseCommand.d.ts +8 -0
- package/dist/makeUseCommand.d.ts.map +1 -0
- package/dist/makeUseCommand.js +13 -0
- package/dist/mutate.d.ts +2 -2
- package/dist/mutate.d.ts.map +1 -1
- package/dist/mutate.js +1 -1
- package/dist/query.d.ts +11 -15
- package/dist/query.d.ts.map +1 -1
- package/dist/query.js +17 -26
- package/dist/routeParams.d.ts +1 -1
- package/dist/runtime.d.ts +5 -2
- package/dist/runtime.d.ts.map +1 -1
- package/dist/runtime.js +27 -17
- package/dist/toast.d.ts +46 -0
- package/dist/toast.d.ts.map +1 -0
- package/dist/toast.js +32 -0
- package/dist/withToast.d.ts +26 -0
- package/dist/withToast.d.ts.map +1 -0
- package/dist/withToast.js +49 -0
- package/eslint.config.mjs +2 -2
- package/package.json +48 -48
- package/src/{experimental/commander.ts → commander.ts} +930 -255
- package/src/{experimental/confirm.ts → confirm.ts} +10 -14
- package/src/errorReporter.ts +60 -72
- package/src/form.ts +55 -16
- package/src/intl.ts +12 -0
- package/src/lib.ts +43 -0
- package/src/makeClient.ts +305 -1015
- package/src/{experimental/makeUseCommand.ts → makeUseCommand.ts} +3 -3
- package/src/query.ts +37 -48
- package/src/runtime.ts +39 -18
- package/src/{experimental/toast.ts → toast.ts} +11 -25
- package/src/{experimental/withToast.ts → withToast.ts} +15 -6
- package/test/Mutation.test.ts +130 -10
- package/test/dist/form.test.d.ts.map +1 -1
- package/test/dist/lib.test.d.ts.map +1 -0
- package/test/dist/stubs.d.ts +1103 -118
- package/test/dist/stubs.d.ts.map +1 -1
- package/test/dist/stubs.js +56 -23
- package/test/form-validation-errors.test.ts +23 -19
- package/test/form.test.ts +20 -2
- package/test/lib.test.ts +240 -0
- package/test/makeClient.test.ts +93 -39
- package/test/stubs.ts +83 -41
- package/tsconfig.json +0 -1
- package/tsconfig.json.bak +2 -2
- package/tsconfig.src.json +34 -34
- package/tsconfig.test.json +2 -2
- package/vitest.config.ts +5 -5
- package/dist/experimental/commander.d.ts +0 -359
- package/dist/experimental/commander.d.ts.map +0 -1
- package/dist/experimental/commander.js +0 -557
- package/dist/experimental/confirm.d.ts +0 -19
- package/dist/experimental/confirm.d.ts.map +0 -1
- package/dist/experimental/confirm.js +0 -28
- package/dist/experimental/intl.d.ts +0 -16
- package/dist/experimental/intl.d.ts.map +0 -1
- package/dist/experimental/intl.js +0 -5
- package/dist/experimental/makeUseCommand.d.ts +0 -8
- package/dist/experimental/makeUseCommand.d.ts.map +0 -1
- package/dist/experimental/makeUseCommand.js +0 -13
- package/dist/experimental/toast.d.ts +0 -47
- package/dist/experimental/toast.d.ts.map +0 -1
- package/dist/experimental/toast.js +0 -41
- package/dist/experimental/withToast.d.ts +0 -25
- package/dist/experimental/withToast.d.ts.map +0 -1
- package/dist/experimental/withToast.js +0 -45
- package/src/experimental/intl.ts +0 -9
package/test/dist/stubs.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"stubs.d.ts","sourceRoot":"","sources":["../stubs.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,KAAK,oBAAoB,EAAE,MAAM,oCAAoC,CAAA;AAE9E,OAAO,EAAU,KAAK,EAA0B,CAAC,EAAE,MAAM,YAAY,CAAA;AACrE,OAAO,EAAE,gBAAgB,EAAiB,MAAM,mBAAmB,CAAA;AACnE,OAAO,EAAE,aAAa,EAAE,MAAM,gBAAgB,CAAA;AAG9C,OAAO,EAAE,SAAS,EAAE,MAAM,
|
|
1
|
+
{"version":3,"file":"stubs.d.ts","sourceRoot":"","sources":["../stubs.ts"],"names":[],"mappings":"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"}
|
package/test/dist/stubs.js
CHANGED
|
@@ -4,12 +4,12 @@ import { ApiClientFactory, makeRpcClient } from "effect-app/client";
|
|
|
4
4
|
import { RpcContextMap } from "effect-app/rpc";
|
|
5
5
|
import * as FetchHttpClient from "effect/unstable/http/FetchHttpClient";
|
|
6
6
|
import { ref } from "vue";
|
|
7
|
-
import { Commander } from "../src/
|
|
8
|
-
import { I18n } from "../src/
|
|
9
|
-
import {
|
|
10
|
-
import
|
|
11
|
-
import
|
|
12
|
-
import {
|
|
7
|
+
import { Commander } from "../src/commander.js";
|
|
8
|
+
import { I18n } from "../src/intl.js";
|
|
9
|
+
import { makeClient } from "../src/makeClient.js";
|
|
10
|
+
import { makeUseCommand } from "../src/makeUseCommand.js";
|
|
11
|
+
import * as Toast from "../src/toast.js";
|
|
12
|
+
import { WithToast } from "../src/withToast.js";
|
|
13
13
|
const fakeToastLayer = (toasts = []) => Layer.effect(Toast.Toast, Effect.sync(() => {
|
|
14
14
|
const dismiss = (id) => {
|
|
15
15
|
const idx = toasts.findIndex((_) => _.id === id);
|
|
@@ -19,21 +19,21 @@ const fakeToastLayer = (toasts = []) => Layer.effect(Toast.Toast, Effect.sync(()
|
|
|
19
19
|
toasts.splice(idx, 1);
|
|
20
20
|
}
|
|
21
21
|
};
|
|
22
|
-
const fakeToast = (message, options) => {
|
|
22
|
+
const fakeToast = (type) => (message, options) => {
|
|
23
23
|
const id = options?.id ?? Math.random().toString(36).substring(2, 15);
|
|
24
|
-
console.log(`Toast [${id}]: ${message}`, options);
|
|
24
|
+
console.log(`Toast [${type}][${id}]: ${message}`, options);
|
|
25
25
|
options = { ...options, id };
|
|
26
26
|
const idx = toasts.findIndex((_) => _.id === id);
|
|
27
27
|
if (idx > -1) {
|
|
28
28
|
const toast = toasts[idx];
|
|
29
29
|
clearTimeout(toast.timeoutId);
|
|
30
|
-
Object.assign(toast, { message, options });
|
|
30
|
+
Object.assign(toast, { type, message, options });
|
|
31
31
|
toast.timeoutId = setTimeout(() => {
|
|
32
32
|
toasts.splice(idx, 1);
|
|
33
33
|
}, options?.timeout ?? 3000);
|
|
34
34
|
}
|
|
35
35
|
else {
|
|
36
|
-
const toast = { id, message, options };
|
|
36
|
+
const toast = { id, type, message, options };
|
|
37
37
|
toast.timeoutId = setTimeout(() => {
|
|
38
38
|
toasts.splice(idx, 1);
|
|
39
39
|
}, options?.timeout ?? 3000);
|
|
@@ -42,10 +42,10 @@ const fakeToastLayer = (toasts = []) => Layer.effect(Toast.Toast, Effect.sync(()
|
|
|
42
42
|
return id;
|
|
43
43
|
};
|
|
44
44
|
return Toast.Toast.of(Toast.wrap({
|
|
45
|
-
error: fakeToast,
|
|
46
|
-
warning: fakeToast,
|
|
47
|
-
success: fakeToast,
|
|
48
|
-
info: fakeToast,
|
|
45
|
+
error: fakeToast("error"),
|
|
46
|
+
warning: fakeToast("warning"),
|
|
47
|
+
success: fakeToast("success"),
|
|
48
|
+
info: fakeToast("info"),
|
|
49
49
|
dismiss
|
|
50
50
|
}));
|
|
51
51
|
}));
|
|
@@ -76,29 +76,62 @@ export const useExperimental = (options) => {
|
|
|
76
76
|
};
|
|
77
77
|
export class RequestContextMap extends RpcContextMap.makeMap({}) {
|
|
78
78
|
}
|
|
79
|
-
export const {
|
|
80
|
-
export
|
|
79
|
+
export const { TaggedRequestFor } = makeRpcClient(RequestContextMap);
|
|
80
|
+
export const SomethingReq = TaggedRequestFor("Something");
|
|
81
|
+
const SomethingQuery = SomethingReq.Query;
|
|
82
|
+
const SomethingCommand = SomethingReq.Command;
|
|
83
|
+
class SomethingGetSomething2 extends SomethingQuery()("GetSomething2", {
|
|
81
84
|
id: S.String
|
|
82
|
-
}, { success: S.
|
|
85
|
+
}, { success: S.FiniteFromString }) {
|
|
83
86
|
}
|
|
84
|
-
|
|
87
|
+
class SomethingGetSomething2WithDependencies extends SomethingQuery()("GetSomething2", {
|
|
85
88
|
id: S.String
|
|
86
89
|
}, {
|
|
87
90
|
// this is intentilally fake, to simulate a codec that requires a dependency
|
|
88
|
-
success: S.
|
|
91
|
+
success: S.FiniteFromString,
|
|
89
92
|
error: S.String
|
|
90
93
|
}) {
|
|
91
94
|
}
|
|
92
|
-
|
|
95
|
+
class SomethingDoSomething extends SomethingCommand()("DoSomething", {
|
|
96
|
+
id: S.String
|
|
97
|
+
}, { success: S.FiniteFromString }) {
|
|
98
|
+
}
|
|
99
|
+
// success schema has encoded shape { a: string | null } — used to test projection constraints
|
|
100
|
+
class SomethingGetStructNullable extends SomethingQuery()("GetStructNullable", {}, {
|
|
101
|
+
success: S.Struct({ a: S.NullOr(S.String) })
|
|
102
|
+
}) {
|
|
103
|
+
}
|
|
104
|
+
export const Something = {
|
|
105
|
+
GetSomething2: SomethingGetSomething2,
|
|
106
|
+
GetSomething2WithDependencies: SomethingGetSomething2WithDependencies,
|
|
107
|
+
DoSomething: SomethingDoSomething,
|
|
108
|
+
GetStructNullable: SomethingGetStructNullable
|
|
109
|
+
};
|
|
110
|
+
export const SomethingElseReq = TaggedRequestFor("SomethingElse");
|
|
111
|
+
const SomethingElseQuery = SomethingElseReq.Query;
|
|
112
|
+
class SomethingElseGetSomething2 extends SomethingElseQuery()("GetSomething2", {
|
|
113
|
+
id: S.String
|
|
114
|
+
}, { success: S.FiniteFromString }) {
|
|
115
|
+
}
|
|
116
|
+
class SomethingElseGetSomething2WithDependencies extends SomethingElseQuery()("GetSomething2", {
|
|
117
|
+
id: S.String
|
|
118
|
+
}, {
|
|
119
|
+
success: S.FiniteFromString,
|
|
120
|
+
error: S.String
|
|
121
|
+
}) {
|
|
122
|
+
}
|
|
123
|
+
export const SomethingElse = {
|
|
124
|
+
GetSomething2: SomethingElseGetSomething2,
|
|
125
|
+
GetSomething2WithDependencies: SomethingElseGetSomething2WithDependencies
|
|
126
|
+
};
|
|
93
127
|
export const useClient = (options) => {
|
|
94
128
|
const FakeIntlLayer = fakeIntlLayer(options?.messages);
|
|
95
129
|
const FakeToastLayer = fakeToastLayer(options?.toasts);
|
|
96
130
|
const CommanderLayer = Commander.Default.pipe(Layer.provide([FakeIntlLayer, FakeToastLayer]));
|
|
97
131
|
const WithToastLayer = WithToast.Default.pipe(Layer.provide(FakeToastLayer));
|
|
98
132
|
const api = ApiClientFactory.layer({ url: "bogus", headers: Option.none() }).pipe(Layer.provide(FetchHttpClient.layer));
|
|
99
|
-
const
|
|
100
|
-
const layers = Layer.mergeAll(CommanderLayer, WithToastLayer, FakeToastLayer, FakeIntlLayer, api, lm);
|
|
133
|
+
const layers = Layer.mergeAll(CommanderLayer, WithToastLayer, FakeToastLayer, FakeIntlLayer, api);
|
|
101
134
|
const clientFor_ = ApiClientFactory.makeFor(Layer.empty);
|
|
102
135
|
return makeClient(() => ManagedRuntime.make(layers), clientFor_, Layer.empty);
|
|
103
136
|
};
|
|
104
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
137
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3R1YnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zdHVicy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxPQUFPLEtBQUssSUFBSSxNQUFNLGdCQUFnQixDQUFBO0FBQ3RDLE9BQU8sRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFLGNBQWMsRUFBRSxNQUFNLEVBQUUsQ0FBQyxFQUFFLE1BQU0sWUFBWSxDQUFBO0FBQ3JFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxhQUFhLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQTtBQUNuRSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sZ0JBQWdCLENBQUE7QUFDOUMsT0FBTyxLQUFLLGVBQWUsTUFBTSxzQ0FBc0MsQ0FBQTtBQUN2RSxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0sS0FBSyxDQUFBO0FBQ3pCLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQTtBQUMvQyxPQUFPLEVBQUUsSUFBSSxFQUFFLE1BQU0sZ0JBQWdCLENBQUE7QUFDckMsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLHNCQUFzQixDQUFBO0FBRWpELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQTtBQUN6RCxPQUFPLEtBQUssS0FBSyxNQUFNLGlCQUFpQixDQUFBO0FBQ3hDLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQTtBQUUvQyxNQUFNLGNBQWMsR0FBRyxDQUFDLE1BQU0sR0FBVSxFQUFFLEVBQUUsRUFBRSxDQUM1QyxLQUFLLENBQUMsTUFBTSxDQUNWLEtBQUssQ0FBQyxLQUFLLEVBQ1gsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLEVBQUU7SUFDZixNQUFNLE9BQU8sR0FBRyxDQUFDLEVBQWlCLEVBQUUsRUFBRTtRQUNwQyxNQUFNLEdBQUcsR0FBRyxNQUFNLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsRUFBRSxLQUFLLEVB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
|
|
@@ -4,7 +4,7 @@ import { buildFieldInfoFromFieldsRoot, translate } from "../src/form.js"
|
|
|
4
4
|
// test schema with integer field
|
|
5
5
|
class TestSchema extends S.Class<TestSchema>("TestSchema")({
|
|
6
6
|
integerField: S.Int,
|
|
7
|
-
numberField: S.
|
|
7
|
+
numberField: S.Finite,
|
|
8
8
|
stringField: S.String
|
|
9
9
|
}) {}
|
|
10
10
|
|
|
@@ -137,22 +137,26 @@ it("validates integer field with valid integer", () =>
|
|
|
137
137
|
.pipe(Effect.runPromise))
|
|
138
138
|
|
|
139
139
|
it("error message format matches regex pattern", () => {
|
|
140
|
-
// test the actual error message format from Effect Schema
|
|
141
|
-
const
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
expect(
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
const
|
|
156
|
-
expect(
|
|
157
|
-
expect(
|
|
140
|
+
// test the actual error message format from Effect Schema (both old "actual" and new "got" formats)
|
|
141
|
+
const errorMessageOld = `Expected an integer, actual 59.5`
|
|
142
|
+
const errorMessageNew = `SchemaError(Expected an integer, got 59.5)`
|
|
143
|
+
|
|
144
|
+
const oldMatch = errorMessageOld.match(/Expected.*integer.*(?:actual|got)\s+([^)]+)/i)
|
|
145
|
+
expect(oldMatch).toBeTruthy()
|
|
146
|
+
expect(oldMatch![1]).toBe("59.5")
|
|
147
|
+
|
|
148
|
+
const newMatch = errorMessageNew.match(/Expected.*integer.*(?:actual|got)\s+([^)]+)/i)
|
|
149
|
+
expect(newMatch).toBeTruthy()
|
|
150
|
+
expect(newMatch![1]).toBe("59.5")
|
|
151
|
+
|
|
152
|
+
const numberErrorOld = `Expected a number, actual "not-a-number"`
|
|
153
|
+
const numberErrorNew = `SchemaError(Expected a finite number, got "not-a-number")`
|
|
154
|
+
|
|
155
|
+
const numOldMatch = numberErrorOld.match(/Expected.*number.*(?:actual|got)\s+([^)]+)/i)
|
|
156
|
+
expect(numOldMatch).toBeTruthy()
|
|
157
|
+
expect(numOldMatch![1]).toBe("\"not-a-number\"")
|
|
158
|
+
|
|
159
|
+
const numNewMatch = numberErrorNew.match(/Expected.*number.*(?:actual|got)\s+([^)]+)/i)
|
|
160
|
+
expect(numNewMatch).toBeTruthy()
|
|
161
|
+
expect(numNewMatch![1]).toBe("\"not-a-number\"")
|
|
158
162
|
})
|
package/test/form.test.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Effect, S } from "effect-app"
|
|
2
|
-
import { buildFieldInfoFromFieldsRoot, type DiscriminatedUnionFieldInfo, type FieldInfo, type NestedFieldInfo, type UnionFieldInfo } from "../src/form.js"
|
|
2
|
+
import { buildFieldInfoFromFieldsRoot, type DiscriminatedUnionFieldInfo, type FieldInfo, getMetadataFromSchema, type NestedFieldInfo, type UnionFieldInfo } from "../src/form.js"
|
|
3
3
|
|
|
4
4
|
export class NestedSchema extends S.Class<NestedSchema>("NestedSchema")({
|
|
5
5
|
shallow: S.String,
|
|
@@ -9,7 +9,11 @@ export class NestedSchema extends S.Class<NestedSchema>("NestedSchema")({
|
|
|
9
9
|
deepest: S.Number
|
|
10
10
|
})
|
|
11
11
|
}),
|
|
12
|
-
age: S.Struct({ nfs: S.NumberFromString.pipe(S.decodeTo(S.PositiveInt)) })
|
|
12
|
+
age: S.Struct({ nfs: S.NumberFromString.pipe(S.decodeTo(S.PositiveInt)) }),
|
|
13
|
+
testNumber: S.Number,
|
|
14
|
+
testFinite: S.Finite,
|
|
15
|
+
testNullableNummber: S.NullOr(S.Number),
|
|
16
|
+
testOptionalNumber: S.optional(S.Number)
|
|
13
17
|
}) {}
|
|
14
18
|
|
|
15
19
|
export class SchemaContainsClass extends S.Class<SchemaContainsClass>("SchemaContainsClass")({
|
|
@@ -156,6 +160,15 @@ function testDiscriminatedUnionFieldInfo<T extends Record<PropertyKey, any>>(duf
|
|
|
156
160
|
)
|
|
157
161
|
}
|
|
158
162
|
|
|
163
|
+
it("getMetadataFromSchema handles composed numeric schemas", () => {
|
|
164
|
+
expect(getMetadataFromSchema(S.Number.ast).type).toBe("float")
|
|
165
|
+
expect(getMetadataFromSchema(S.Finite.ast).type).toBe("float")
|
|
166
|
+
expect(getMetadataFromSchema(S.PositiveNumber.ast).type).toBe("float")
|
|
167
|
+
expect(getMetadataFromSchema(S.Int.ast).type).toBe("int")
|
|
168
|
+
expect(getMetadataFromSchema(S.PositiveInt.ast).type).toBe("int")
|
|
169
|
+
expect(getMetadataFromSchema(S.NullOr(S.Number).ast).type).toBe("float")
|
|
170
|
+
})
|
|
171
|
+
|
|
159
172
|
it("buildFieldInfo", () =>
|
|
160
173
|
Effect
|
|
161
174
|
.gen(function*() {
|
|
@@ -176,6 +189,11 @@ it("buildFieldInfo", () =>
|
|
|
176
189
|
testNestedFieldInfo(nestedFieldinfo)
|
|
177
190
|
testNestedFieldInfo(nestedFieldinfo.fields.nested)
|
|
178
191
|
testNestedFieldInfo(nestedFieldinfo.fields.age)
|
|
192
|
+
|
|
193
|
+
expect(nestedFieldinfo.fields.testNumber.type).toBe("float")
|
|
194
|
+
expect(nestedFieldinfo.fields.testFinite.type).toBe("float")
|
|
195
|
+
expect(nestedFieldinfo.fields.testNullableNummber.type).toBe("float")
|
|
196
|
+
expect(nestedFieldinfo.fields.testOptionalNumber.type).toBe("float")
|
|
179
197
|
})
|
|
180
198
|
.pipe(Effect.runPromise))
|
|
181
199
|
|
package/test/lib.test.ts
ADDED
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest"
|
|
2
|
+
import { computed, isProxy, isReactive, isRef, reactive, ref } from "vue"
|
|
3
|
+
import { deepToRaw } from "../src/lib.js"
|
|
4
|
+
|
|
5
|
+
type DeepMapKey = { id: string } | "list"
|
|
6
|
+
type DeepMapValue = { nestedSet: Set<{ ok: boolean } | Date> } | Array<{ count: number }>
|
|
7
|
+
type DeepSetValue = Map<string, { value: number }> | Array<{ value: number }>
|
|
8
|
+
|
|
9
|
+
const expectPlainDeep = (value: unknown): void => {
|
|
10
|
+
expect(isRef(value)).toBe(false)
|
|
11
|
+
expect(isReactive(value)).toBe(false)
|
|
12
|
+
expect(isProxy(value)).toBe(false)
|
|
13
|
+
|
|
14
|
+
if (Array.isArray(value)) {
|
|
15
|
+
value.forEach(expectPlainDeep)
|
|
16
|
+
return
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
if (value instanceof Map) {
|
|
20
|
+
value.forEach((entryValue, entryKey) => {
|
|
21
|
+
expectPlainDeep(entryKey)
|
|
22
|
+
expectPlainDeep(entryValue)
|
|
23
|
+
})
|
|
24
|
+
return
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
if (value instanceof Set) {
|
|
28
|
+
value.forEach((entry) => {
|
|
29
|
+
expectPlainDeep(entry)
|
|
30
|
+
})
|
|
31
|
+
return
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
if (value instanceof Date) {
|
|
35
|
+
return
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
if (value && typeof value === "object") {
|
|
39
|
+
Object.values(value).forEach(expectPlainDeep)
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
describe("deepToRaw", () => {
|
|
44
|
+
it("supports non-object root inputs", () => {
|
|
45
|
+
expect(deepToRaw(1)).toBe(1)
|
|
46
|
+
expect(deepToRaw("x")).toBe("x")
|
|
47
|
+
expect(deepToRaw(null)).toBe(null)
|
|
48
|
+
expect(deepToRaw(undefined)).toBe(undefined)
|
|
49
|
+
expect(deepToRaw(ref(123))).toBe(123)
|
|
50
|
+
|
|
51
|
+
const rootArray = deepToRaw(reactive([reactive({ n: 1 }), ref(2)]))
|
|
52
|
+
expect(rootArray).toEqual([{ n: 1 }, 2])
|
|
53
|
+
expect(Array.isArray(rootArray)).toBe(true)
|
|
54
|
+
|
|
55
|
+
const rootMap = deepToRaw(
|
|
56
|
+
reactive(new Map<string, unknown>([["k", reactive({ n: 1 })], ["r", ref(2)]]))
|
|
57
|
+
)
|
|
58
|
+
expect(rootMap).toBeInstanceOf(Map)
|
|
59
|
+
expect(rootMap.get("k")).toEqual({ n: 1 })
|
|
60
|
+
expect(rootMap.get("r")).toBe(2)
|
|
61
|
+
|
|
62
|
+
const rootSet = deepToRaw(reactive(new Set([reactive({ n: 1 }), ref(2)])))
|
|
63
|
+
expect(rootSet).toBeInstanceOf(Set)
|
|
64
|
+
expect(Array.from(rootSet)).toEqual([{ n: 1 }, 2])
|
|
65
|
+
|
|
66
|
+
const date = new Date("2024-02-03T00:00:00.000Z")
|
|
67
|
+
const rootDate = deepToRaw(date)
|
|
68
|
+
expect(rootDate).toBeInstanceOf(Date)
|
|
69
|
+
expect(rootDate).not.toBe(date)
|
|
70
|
+
expect(rootDate.toISOString()).toBe(date.toISOString())
|
|
71
|
+
})
|
|
72
|
+
|
|
73
|
+
it("unwraps nested objects and arrays without leaving vue proxies behind", () => {
|
|
74
|
+
const source = reactive({
|
|
75
|
+
list: [
|
|
76
|
+
reactive({
|
|
77
|
+
nested: reactive({
|
|
78
|
+
count: 1,
|
|
79
|
+
items: [reactive({ label: "a" }), reactive({ label: "b" })]
|
|
80
|
+
})
|
|
81
|
+
})
|
|
82
|
+
],
|
|
83
|
+
plain: reactive({ ok: true })
|
|
84
|
+
})
|
|
85
|
+
|
|
86
|
+
const result = deepToRaw(source)
|
|
87
|
+
|
|
88
|
+
expect(Array.isArray(result.list)).toBe(true)
|
|
89
|
+
expect(Array.isArray(result.list[0]?.nested.items)).toBe(true)
|
|
90
|
+
expect(result).toEqual({
|
|
91
|
+
list: [{ nested: { count: 1, items: [{ label: "a" }, { label: "b" }] } }],
|
|
92
|
+
plain: { ok: true }
|
|
93
|
+
})
|
|
94
|
+
expectPlainDeep(result)
|
|
95
|
+
})
|
|
96
|
+
|
|
97
|
+
it("preserves maps and sets while deeply unwrapping nested entries", () => {
|
|
98
|
+
const key = reactive({ id: "key" })
|
|
99
|
+
const nestedDate = new Date("2024-01-02T03:04:05.000Z")
|
|
100
|
+
const map = reactive(
|
|
101
|
+
new Map<DeepMapKey, DeepMapValue>([
|
|
102
|
+
[key, reactive({ nestedSet: reactive(new Set([{ ok: true }, nestedDate])) })],
|
|
103
|
+
["list", reactive([{ count: 2 }])]
|
|
104
|
+
])
|
|
105
|
+
)
|
|
106
|
+
const set = reactive(
|
|
107
|
+
new Set<DeepSetValue>([
|
|
108
|
+
reactive(new Map([["deep", reactive({ value: 3 })]])),
|
|
109
|
+
reactive([{ value: 4 }])
|
|
110
|
+
])
|
|
111
|
+
)
|
|
112
|
+
const source = reactive({
|
|
113
|
+
map,
|
|
114
|
+
set
|
|
115
|
+
})
|
|
116
|
+
|
|
117
|
+
const result = deepToRaw(source)
|
|
118
|
+
|
|
119
|
+
expect(result.map).toBeInstanceOf(Map)
|
|
120
|
+
expect(result.set).toBeInstanceOf(Set)
|
|
121
|
+
|
|
122
|
+
const entries = Array.from(result.map.entries())
|
|
123
|
+
expect(entries[0]?.[0]).toEqual({ id: "key" })
|
|
124
|
+
expect(entries[0]?.[0]).not.toBe(key)
|
|
125
|
+
expect(entries[0]?.[1]).toEqual({ nestedSet: new Set([{ ok: true }, nestedDate]) })
|
|
126
|
+
expect(entries[1]?.[1]).toEqual([{ count: 2 }])
|
|
127
|
+
|
|
128
|
+
const setValues = Array.from(result.set.values())
|
|
129
|
+
expect(setValues[0]).toBeInstanceOf(Map)
|
|
130
|
+
expect(setValues[1]).toEqual([{ value: 4 }])
|
|
131
|
+
expect((setValues[0] as Map<string, { value: number }>).get("deep")).toEqual({ value: 3 })
|
|
132
|
+
|
|
133
|
+
expectPlainDeep(result)
|
|
134
|
+
})
|
|
135
|
+
|
|
136
|
+
it("keeps nested dates as dates, including dates reached through refs", () => {
|
|
137
|
+
const date = new Date("2025-06-07T08:09:10.000Z")
|
|
138
|
+
const source = reactive({
|
|
139
|
+
createdAt: date,
|
|
140
|
+
nested: reactive({
|
|
141
|
+
updatedAt: ref(date),
|
|
142
|
+
list: [ref(date)],
|
|
143
|
+
map: reactive(new Map([["at", ref(date)]])),
|
|
144
|
+
set: reactive(new Set([ref(date)]))
|
|
145
|
+
})
|
|
146
|
+
})
|
|
147
|
+
|
|
148
|
+
const result = deepToRaw(source)
|
|
149
|
+
|
|
150
|
+
expect(result.createdAt).toBeInstanceOf(Date)
|
|
151
|
+
expect(result.nested.updatedAt).toBeInstanceOf(Date)
|
|
152
|
+
expect(result.nested.list[0]).toBeInstanceOf(Date)
|
|
153
|
+
expect(result.nested.map).toBeInstanceOf(Map)
|
|
154
|
+
expect(result.nested.set).toBeInstanceOf(Set)
|
|
155
|
+
|
|
156
|
+
const updatedAt = result.nested.updatedAt
|
|
157
|
+
const firstListDate = result.nested.list[0]
|
|
158
|
+
const mappedDate = result.nested.map.get("at")
|
|
159
|
+
const firstSetDate = Array.from(result.nested.set)[0]
|
|
160
|
+
|
|
161
|
+
if (!(updatedAt instanceof Date)) {
|
|
162
|
+
throw new Error("expected updatedAt to be a Date")
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
if (!(firstListDate instanceof Date)) {
|
|
166
|
+
throw new Error("expected first list item to be a Date")
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
if (!(mappedDate instanceof Date)) {
|
|
170
|
+
throw new Error("expected mapped date to be a Date")
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
if (!(firstSetDate instanceof Date)) {
|
|
174
|
+
throw new Error("expected first set item to be a Date")
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
expect(result.createdAt.toISOString()).toBe(date.toISOString())
|
|
178
|
+
expect(updatedAt.toISOString()).toBe(date.toISOString())
|
|
179
|
+
expect(firstListDate.toISOString()).toBe(date.toISOString())
|
|
180
|
+
expect(mappedDate.toISOString()).toBe(date.toISOString())
|
|
181
|
+
expect(firstSetDate.toISOString()).toBe(date.toISOString())
|
|
182
|
+
|
|
183
|
+
expectPlainDeep(result)
|
|
184
|
+
})
|
|
185
|
+
|
|
186
|
+
it("unwraps computed values nested in refs/plain objects and deepToRawes the computed result", () => {
|
|
187
|
+
const source = {
|
|
188
|
+
innerRef: ref({
|
|
189
|
+
computedValue: computed(() =>
|
|
190
|
+
reactive({
|
|
191
|
+
list: [reactive({ n: 1 }), reactive({ n: 2 })],
|
|
192
|
+
map: reactive(new Map([["k", reactive({ nested: true })]])),
|
|
193
|
+
set: reactive(new Set([reactive({ fromSet: true })]))
|
|
194
|
+
})
|
|
195
|
+
)
|
|
196
|
+
}),
|
|
197
|
+
plainComputed: computed(() => reactive({ date: ref(new Date("2025-01-01T00:00:00.000Z")) }))
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
const result = deepToRaw(source)
|
|
201
|
+
|
|
202
|
+
expect(result).toEqual({
|
|
203
|
+
innerRef: {
|
|
204
|
+
computedValue: {
|
|
205
|
+
list: [{ n: 1 }, { n: 2 }],
|
|
206
|
+
map: new Map([["k", { nested: true }]]),
|
|
207
|
+
set: new Set([{ fromSet: true }])
|
|
208
|
+
}
|
|
209
|
+
},
|
|
210
|
+
plainComputed: {
|
|
211
|
+
date: new Date("2025-01-01T00:00:00.000Z")
|
|
212
|
+
}
|
|
213
|
+
})
|
|
214
|
+
|
|
215
|
+
const innerRefValue = Reflect.get(result, "innerRef")
|
|
216
|
+
if (!innerRefValue || typeof innerRefValue !== "object") {
|
|
217
|
+
throw new Error("expected innerRef to be an object")
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
const computedValue = Reflect.get(innerRefValue, "computedValue")
|
|
221
|
+
if (!computedValue || typeof computedValue !== "object") {
|
|
222
|
+
throw new Error("expected computedValue to be an object")
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
const computedMap = Reflect.get(computedValue, "map")
|
|
226
|
+
const computedSet = Reflect.get(computedValue, "set")
|
|
227
|
+
|
|
228
|
+
const plainComputedValue = Reflect.get(result, "plainComputed")
|
|
229
|
+
if (!plainComputedValue || typeof plainComputedValue !== "object") {
|
|
230
|
+
throw new Error("expected plainComputed to be an object")
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
const computedDate = Reflect.get(plainComputedValue, "date")
|
|
234
|
+
|
|
235
|
+
expect(computedMap).toBeInstanceOf(Map)
|
|
236
|
+
expect(computedSet).toBeInstanceOf(Set)
|
|
237
|
+
expect(computedDate).toBeInstanceOf(Date)
|
|
238
|
+
expectPlainDeep(result)
|
|
239
|
+
})
|
|
240
|
+
})
|