sonamu 0.8.24 → 0.8.26
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/api/__tests__/config.test.js +189 -0
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +7 -2
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +14 -10
- package/dist/auth/index.d.ts +1 -0
- package/dist/auth/index.d.ts.map +1 -1
- package/dist/auth/index.js +2 -1
- package/dist/auth/knex-adapter.d.ts +23 -0
- package/dist/auth/knex-adapter.d.ts.map +1 -0
- package/dist/auth/knex-adapter.js +163 -0
- package/dist/auth/plugins/wrappers/admin.d.ts +2 -2
- package/dist/bin/__tests__/ts-loader-register.test.js +45 -0
- package/dist/bin/cli.js +47 -9
- package/dist/bin/ts-loader-register.js +3 -29
- package/dist/bin/ts-loader-registration.d.ts +2 -0
- package/dist/bin/ts-loader-registration.d.ts.map +1 -0
- package/dist/bin/ts-loader-registration.js +42 -0
- package/dist/cone/cone-generator.js +3 -3
- package/dist/database/puri-subset.test-d.js +9 -1
- package/dist/database/puri-subset.types.d.ts +1 -1
- package/dist/database/puri-subset.types.d.ts.map +1 -1
- package/dist/database/puri-subset.types.js +1 -1
- package/dist/testing/fixture-generator.js +5 -5
- package/dist/ui/ai-client.js +2 -2
- package/dist/ui/api.d.ts.map +1 -1
- package/dist/ui/api.js +14 -14
- package/dist/ui/cdd-service.d.ts +15 -18
- package/dist/ui/cdd-service.d.ts.map +1 -1
- package/dist/ui/cdd-service.js +246 -222
- package/dist/ui/cdd-types.d.ts +41 -68
- package/dist/ui/cdd-types.d.ts.map +1 -1
- package/dist/ui/cdd-types.js +2 -2
- package/dist/ui-web/assets/index-CKo0Z2Iu.css +1 -0
- package/dist/ui-web/assets/{index-CxiydzeC.js → index-DK-2aacv.js} +83 -83
- package/dist/ui-web/index.html +2 -2
- package/package.json +6 -2
- package/src/api/__tests__/config.test.ts +225 -0
- package/src/api/config.ts +10 -4
- package/src/api/sonamu.ts +16 -13
- package/src/auth/index.ts +1 -0
- package/src/auth/knex-adapter.ts +208 -0
- package/src/bin/__tests__/ts-loader-register.test.ts +62 -0
- package/src/bin/cli.ts +52 -9
- package/src/bin/ts-loader-register.ts +2 -32
- package/src/bin/ts-loader-registration.ts +55 -0
- package/src/cone/cone-generator.ts +2 -2
- package/src/database/puri-subset.test-d.ts +102 -0
- package/src/database/puri-subset.types.ts +1 -1
- package/src/skills/commands/sonamu-skills.md +20 -0
- package/src/skills/sonamu/SKILL.md +179 -137
- package/src/skills/sonamu/ai-agents.md +69 -69
- package/src/skills/sonamu/api.md +147 -147
- package/src/skills/sonamu/auth-migration.md +220 -220
- package/src/skills/sonamu/auth-plugins.md +83 -83
- package/src/skills/sonamu/auth.md +106 -106
- package/src/skills/sonamu/cdd.md +65 -200
- package/src/skills/sonamu/cone.md +138 -138
- package/src/skills/sonamu/config.md +191 -191
- package/src/skills/sonamu/create-sonamu.md +66 -66
- package/src/skills/sonamu/database.md +158 -158
- package/src/skills/sonamu/entity-basic.md +292 -293
- package/src/skills/sonamu/entity-relations.md +246 -246
- package/src/skills/sonamu/entity-validation-checklist.md +124 -124
- package/src/skills/sonamu/fixture-cli.md +231 -231
- package/src/skills/sonamu/framework-change.md +37 -37
- package/src/skills/sonamu/frontend.md +223 -223
- package/src/skills/sonamu/i18n.md +82 -82
- package/src/skills/sonamu/migration.md +77 -77
- package/src/skills/sonamu/model.md +222 -222
- package/src/skills/sonamu/naite.md +86 -86
- package/src/skills/sonamu/project-init.md +228 -228
- package/src/skills/sonamu/puri.md +122 -122
- package/src/skills/sonamu/scaffolding.md +154 -154
- package/src/skills/sonamu/skill-contribution.md +124 -124
- package/src/skills/sonamu/subset.md +46 -46
- package/src/skills/sonamu/tasks.md +82 -82
- package/src/skills/sonamu/testing-devrunner.md +147 -147
- package/src/skills/sonamu/testing.md +673 -673
- package/src/skills/sonamu/upsert.md +79 -79
- package/src/skills/sonamu/vector.md +67 -67
- package/src/testing/fixture-generator.ts +4 -4
- package/src/ui/ai-client.ts +1 -1
- package/src/ui/api.ts +18 -17
- package/src/ui/cdd-service.ts +264 -254
- package/src/ui/cdd-types.ts +40 -75
- package/dist/ui-web/assets/index-BrQKU3j9.css +0 -1
- package/src/skills/sonamu/workflow.md +0 -317
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import type { BetterAuthOptions } from "better-auth";
|
|
2
|
+
import type { Knex } from "knex";
|
|
3
|
+
interface CleanedWhere {
|
|
4
|
+
field: string;
|
|
5
|
+
value: string | number | boolean | string[] | number[] | Date | null;
|
|
6
|
+
operator: string;
|
|
7
|
+
connector: string;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* better-auth용 Sonamu knex 어댑터
|
|
11
|
+
*
|
|
12
|
+
* better-auth의 모든 쿼리를 DB.getDB()를 통해 실행하여
|
|
13
|
+
* Sonamu 테스트 트랜잭션과 동일한 커넥션을 공유합니다.
|
|
14
|
+
*/
|
|
15
|
+
export declare const sonamuKnexAdapter: () => (options: BetterAuthOptions) => import("better-auth").DBAdapter<BetterAuthOptions>;
|
|
16
|
+
/**
|
|
17
|
+
* Better Auth의 공식 어댑터(Kysely, Drizzle, Prisma, MongoDB) 패턴에 맞춰
|
|
18
|
+
* AND 그룹과 OR 그룹을 분리한 뒤 top-level AND로 결합합니다.
|
|
19
|
+
* 결과: (A AND B AND ...) AND (C OR D OR ...)
|
|
20
|
+
*/
|
|
21
|
+
export declare function applyWhere(query: Knex.QueryBuilder, conditions: CleanedWhere[]): Knex.QueryBuilder;
|
|
22
|
+
export {};
|
|
23
|
+
//# sourceMappingURL=knex-adapter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"knex-adapter.d.ts","sourceRoot":"","sources":["../../src/auth/knex-adapter.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAMrD,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAGjC,UAAU,YAAY;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,MAAM,EAAE,GAAG,MAAM,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC;IACrE,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;;;;GAKG;AACH,eAAO,MAAM,iBAAiB,SAyGpB,SAAS,iBAAiB,uDAInC,CAAC;AAEF;;;;GAIG;AACH,wBAAgB,UAAU,CACxB,KAAK,EAAE,IAAI,CAAC,YAAY,EACxB,UAAU,EAAE,YAAY,EAAE,GACzB,IAAI,CAAC,YAAY,CAmBnB"}
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
import { createAdapterFactory } from "better-auth/adapters";
|
|
2
|
+
import { DB } from "../database/db.js";
|
|
3
|
+
/**
|
|
4
|
+
* better-auth용 Sonamu knex 어댑터
|
|
5
|
+
*
|
|
6
|
+
* better-auth의 모든 쿼리를 DB.getDB()를 통해 실행하여
|
|
7
|
+
* Sonamu 테스트 트랜잭션과 동일한 커넥션을 공유합니다.
|
|
8
|
+
*/ export const sonamuKnexAdapter = ()=>{
|
|
9
|
+
let lazyOptions = null;
|
|
10
|
+
const createCustomAdapter = (getDb)=>{
|
|
11
|
+
return ({ getFieldName })=>({
|
|
12
|
+
create: async ({ model, data })=>{
|
|
13
|
+
const [row] = await getDb()(model).insert(data).returning("*");
|
|
14
|
+
return row;
|
|
15
|
+
},
|
|
16
|
+
findOne: async ({ model, where })=>{
|
|
17
|
+
let query = getDb()(model);
|
|
18
|
+
query = applyWhere(query, where);
|
|
19
|
+
const row = await query.first();
|
|
20
|
+
return row ?? null;
|
|
21
|
+
},
|
|
22
|
+
findMany: async ({ model, where, limit, offset, sortBy })=>{
|
|
23
|
+
let query = getDb()(model);
|
|
24
|
+
if (where) {
|
|
25
|
+
query = applyWhere(query, where);
|
|
26
|
+
}
|
|
27
|
+
if (sortBy) {
|
|
28
|
+
const dbField = getFieldName({
|
|
29
|
+
model,
|
|
30
|
+
field: sortBy.field
|
|
31
|
+
});
|
|
32
|
+
query = query.orderBy(dbField, sortBy.direction);
|
|
33
|
+
}
|
|
34
|
+
if (limit) {
|
|
35
|
+
query = query.limit(limit);
|
|
36
|
+
}
|
|
37
|
+
if (offset) {
|
|
38
|
+
query = query.offset(offset);
|
|
39
|
+
}
|
|
40
|
+
return await query;
|
|
41
|
+
},
|
|
42
|
+
update: async ({ model, where, update })=>{
|
|
43
|
+
let query = getDb()(model);
|
|
44
|
+
query = applyWhere(query, where);
|
|
45
|
+
const [row] = await query.update(update).returning("*");
|
|
46
|
+
return row ?? null;
|
|
47
|
+
},
|
|
48
|
+
updateMany: async ({ model, where, update })=>{
|
|
49
|
+
let query = getDb()(model);
|
|
50
|
+
query = applyWhere(query, where);
|
|
51
|
+
const count = await query.update(update);
|
|
52
|
+
return count;
|
|
53
|
+
},
|
|
54
|
+
delete: async ({ model, where })=>{
|
|
55
|
+
let query = getDb()(model);
|
|
56
|
+
query = applyWhere(query, where);
|
|
57
|
+
await query.del();
|
|
58
|
+
},
|
|
59
|
+
deleteMany: async ({ model, where })=>{
|
|
60
|
+
let query = getDb()(model);
|
|
61
|
+
query = applyWhere(query, where);
|
|
62
|
+
const count = await query.del();
|
|
63
|
+
return count;
|
|
64
|
+
},
|
|
65
|
+
count: async ({ model, where })=>{
|
|
66
|
+
let query = getDb()(model);
|
|
67
|
+
if (where) {
|
|
68
|
+
query = applyWhere(query, where);
|
|
69
|
+
}
|
|
70
|
+
const [{ count }] = await query.count("* as count");
|
|
71
|
+
return Number(count);
|
|
72
|
+
}
|
|
73
|
+
});
|
|
74
|
+
};
|
|
75
|
+
const adapterConfig = {
|
|
76
|
+
adapterId: "sonamu-knex",
|
|
77
|
+
adapterName: "Sonamu Knex Adapter",
|
|
78
|
+
usePlural: false,
|
|
79
|
+
supportsJSON: true,
|
|
80
|
+
supportsDates: true,
|
|
81
|
+
supportsBooleans: true,
|
|
82
|
+
supportsNumericIds: false,
|
|
83
|
+
transaction: async (cb)=>{
|
|
84
|
+
const db = DB.getDB("w");
|
|
85
|
+
return db.transaction(async (trx)=>{
|
|
86
|
+
const options = lazyOptions;
|
|
87
|
+
if (!options) {
|
|
88
|
+
throw new Error("sonamuKnexAdapter: options not initialized");
|
|
89
|
+
}
|
|
90
|
+
return cb(createAdapterFactory({
|
|
91
|
+
config: adapterConfig,
|
|
92
|
+
adapter: createCustomAdapter(()=>trx)
|
|
93
|
+
})(options));
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
};
|
|
97
|
+
const adapterCreator = createAdapterFactory({
|
|
98
|
+
config: adapterConfig,
|
|
99
|
+
adapter: createCustomAdapter(()=>DB.getDB("w"))
|
|
100
|
+
});
|
|
101
|
+
return (options)=>{
|
|
102
|
+
lazyOptions = options;
|
|
103
|
+
return adapterCreator(options);
|
|
104
|
+
};
|
|
105
|
+
};
|
|
106
|
+
/**
|
|
107
|
+
* Better Auth의 공식 어댑터(Kysely, Drizzle, Prisma, MongoDB) 패턴에 맞춰
|
|
108
|
+
* AND 그룹과 OR 그룹을 분리한 뒤 top-level AND로 결합합니다.
|
|
109
|
+
* 결과: (A AND B AND ...) AND (C OR D OR ...)
|
|
110
|
+
*/ export function applyWhere(query, conditions) {
|
|
111
|
+
const andGroup = conditions.filter((c)=>c.connector !== "OR");
|
|
112
|
+
const orGroup = conditions.filter((c)=>c.connector === "OR");
|
|
113
|
+
if (andGroup.length > 0) {
|
|
114
|
+
for (const condition of andGroup){
|
|
115
|
+
query = applyCondition(query, condition, "where");
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
if (orGroup.length > 0) {
|
|
119
|
+
query = query.where(function() {
|
|
120
|
+
for(let i = 0; i < orGroup.length; i++){
|
|
121
|
+
applyCondition(this, orGroup[i], i === 0 ? "where" : "orWhere");
|
|
122
|
+
}
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
return query;
|
|
126
|
+
}
|
|
127
|
+
function applyCondition(query, condition, method) {
|
|
128
|
+
const { field, value, operator } = condition;
|
|
129
|
+
switch(operator){
|
|
130
|
+
case "eq":
|
|
131
|
+
if (value === null) {
|
|
132
|
+
return query[method === "orWhere" ? "orWhereNull" : "whereNull"](field);
|
|
133
|
+
}
|
|
134
|
+
return query[method](field, "=", value);
|
|
135
|
+
case "ne":
|
|
136
|
+
if (value === null) {
|
|
137
|
+
return query[method === "orWhere" ? "orWhereNotNull" : "whereNotNull"](field);
|
|
138
|
+
}
|
|
139
|
+
return query[method](field, "!=", value);
|
|
140
|
+
case "lt":
|
|
141
|
+
return query[method](field, "<", value);
|
|
142
|
+
case "lte":
|
|
143
|
+
return query[method](field, "<=", value);
|
|
144
|
+
case "gt":
|
|
145
|
+
return query[method](field, ">", value);
|
|
146
|
+
case "gte":
|
|
147
|
+
return query[method](field, ">=", value);
|
|
148
|
+
case "in":
|
|
149
|
+
return query[method === "orWhere" ? "orWhereIn" : "whereIn"](field, value);
|
|
150
|
+
case "not_in":
|
|
151
|
+
return query[method === "orWhere" ? "orWhereNotIn" : "whereNotIn"](field, value);
|
|
152
|
+
case "contains":
|
|
153
|
+
return query[method](field, "like", `%${value}%`);
|
|
154
|
+
case "starts_with":
|
|
155
|
+
return query[method](field, "like", `${value}%`);
|
|
156
|
+
case "ends_with":
|
|
157
|
+
return query[method](field, "like", `%${value}`);
|
|
158
|
+
default:
|
|
159
|
+
return query;
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9hdXRoL2tuZXgtYWRhcHRlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IEJldHRlckF1dGhPcHRpb25zIH0gZnJvbSBcImJldHRlci1hdXRoXCI7XG5pbXBvcnQgdHlwZSB7XG4gIEFkYXB0ZXJGYWN0b3J5Q3VzdG9taXplQWRhcHRlckNyZWF0b3IsXG4gIERCVHJhbnNhY3Rpb25BZGFwdGVyLFxufSBmcm9tIFwiYmV0dGVyLWF1dGgvYWRhcHRlcnNcIjtcbmltcG9ydCB7IGNyZWF0ZUFkYXB0ZXJGYWN0b3J5IH0gZnJvbSBcImJldHRlci1hdXRoL2FkYXB0ZXJzXCI7XG5pbXBvcnQgdHlwZSB7IEtuZXggfSBmcm9tIFwia25leFwiO1xuaW1wb3J0IHsgREIgfSBmcm9tIFwiLi4vZGF0YWJhc2UvZGJcIjtcblxuaW50ZXJmYWNlIENsZWFuZWRXaGVyZSB7XG4gIGZpZWxkOiBzdHJpbmc7XG4gIHZhbHVlOiBzdHJpbmcgfCBudW1iZXIgfCBib29sZWFuIHwgc3RyaW5nW10gfCBudW1iZXJbXSB8IERhdGUgfCBudWxsO1xuICBvcGVyYXRvcjogc3RyaW5nO1xuICBjb25uZWN0b3I6IHN0cmluZztcbn1cblxuLyoqXG4gKiBiZXR0ZXItYXV0aOyaqSBTb25hbXUga25leCDslrTrjJHthLBcbiAqXG4gKiBiZXR0ZXItYXV0aOydmCDrqqjrk6Ag7L+866as66W8IERCLmdldERCKCnrpbwg7Ya17ZW0IOyLpO2Wie2VmOyXrFxuICogU29uYW11IO2FjOyKpO2KuCDtirjrnpzsnq3shZjqs7wg64+Z7J287ZWcIOy7pOuEpeyFmOydhCDqs7XsnKDtlanri4jri6QuXG4gKi9cbmV4cG9ydCBjb25zdCBzb25hbXVLbmV4QWRhcHRlciA9ICgpID0+IHtcbiAgbGV0IGxhenlPcHRpb25zOiBCZXR0ZXJBdXRoT3B0aW9ucyB8IG51bGwgPSBudWxsO1xuXG4gIGNvbnN0IGNyZWF0ZUN1c3RvbUFkYXB0ZXIgPSAoXG4gICAgZ2V0RGI6ICgpID0+IEtuZXggfCBLbmV4LlRyYW5zYWN0aW9uLFxuICApOiBBZGFwdGVyRmFjdG9yeUN1c3RvbWl6ZUFkYXB0ZXJDcmVhdG9yID0+IHtcbiAgICByZXR1cm4gKHsgZ2V0RmllbGROYW1lIH0pID0+ICh7XG4gICAgICBjcmVhdGU6IGFzeW5jICh7IG1vZGVsLCBkYXRhIH0pID0+IHtcbiAgICAgICAgY29uc3QgW3Jvd10gPSBhd2FpdCBnZXREYigpKG1vZGVsKS5pbnNlcnQoZGF0YSkucmV0dXJuaW5nKFwiKlwiKTtcbiAgICAgICAgcmV0dXJuIHJvdztcbiAgICAgIH0sXG5cbiAgICAgIGZpbmRPbmU6IGFzeW5jICh7IG1vZGVsLCB3aGVyZSB9KSA9PiB7XG4gICAgICAgIGxldCBxdWVyeSA9IGdldERiKCkobW9kZWwpO1xuICAgICAgICBxdWVyeSA9IGFwcGx5V2hlcmUocXVlcnksIHdoZXJlKTtcbiAgICAgICAgY29uc3Qgcm93ID0gYXdhaXQgcXVlcnkuZmlyc3QoKTtcbiAgICAgICAgcmV0dXJuIHJvdyA/PyBudWxsO1xuICAgICAgfSxcblxuICAgICAgZmluZE1hbnk6IGFzeW5jICh7IG1vZGVsLCB3aGVyZSwgbGltaXQsIG9mZnNldCwgc29ydEJ5IH0pID0+IHtcbiAgICAgICAgbGV0IHF1ZXJ5ID0gZ2V0RGIoKShtb2RlbCk7XG4gICAgICAgIGlmICh3aGVyZSkge1xuICAgICAgICAgIHF1ZXJ5ID0gYXBwbHlXaGVyZShxdWVyeSwgd2hlcmUpO1xuICAgICAgICB9XG4gICAgICAgIGlmIChzb3J0QnkpIHtcbiAgICAgICAgICBjb25zdCBkYkZpZWxkID0gZ2V0RmllbGROYW1lKHsgbW9kZWwsIGZpZWxkOiBzb3J0QnkuZmllbGQgfSk7XG4gICAgICAgICAgcXVlcnkgPSBxdWVyeS5vcmRlckJ5KGRiRmllbGQsIHNvcnRCeS5kaXJlY3Rpb24pO1xuICAgICAgICB9XG4gICAgICAgIGlmIChsaW1pdCkge1xuICAgICAgICAgIHF1ZXJ5ID0gcXVlcnkubGltaXQobGltaXQpO1xuICAgICAgICB9XG4gICAgICAgIGlmIChvZmZzZXQpIHtcbiAgICAgICAgICBxdWVyeSA9IHF1ZXJ5Lm9mZnNldChvZmZzZXQpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBhd2FpdCBxdWVyeTtcbiAgICAgIH0sXG5cbiAgICAgIHVwZGF0ZTogYXN5bmMgKHsgbW9kZWwsIHdoZXJlLCB1cGRhdGUgfSkgPT4ge1xuICAgICAgICBsZXQgcXVlcnkgPSBnZXREYigpKG1vZGVsKTtcbiAgICAgICAgcXVlcnkgPSBhcHBseVdoZXJlKHF1ZXJ5LCB3aGVyZSk7XG4gICAgICAgIGNvbnN0IFtyb3ddID0gYXdhaXQgcXVlcnkudXBkYXRlKHVwZGF0ZSkucmV0dXJuaW5nKFwiKlwiKTtcbiAgICAgICAgcmV0dXJuIHJvdyA/PyBudWxsO1xuICAgICAgfSxcblxuICAgICAgdXBkYXRlTWFueTogYXN5bmMgKHsgbW9kZWwsIHdoZXJlLCB1cGRhdGUgfSkgPT4ge1xuICAgICAgICBsZXQgcXVlcnkgPSBnZXREYigpKG1vZGVsKTtcbiAgICAgICAgcXVlcnkgPSBhcHBseVdoZXJlKHF1ZXJ5LCB3aGVyZSk7XG4gICAgICAgIGNvbnN0IGNvdW50ID0gYXdhaXQgcXVlcnkudXBkYXRlKHVwZGF0ZSk7XG4gICAgICAgIHJldHVybiBjb3VudDtcbiAgICAgIH0sXG5cbiAgICAgIGRlbGV0ZTogYXN5bmMgKHsgbW9kZWwsIHdoZXJlIH0pID0+IHtcbiAgICAgICAgbGV0IHF1ZXJ5ID0gZ2V0RGIoKShtb2RlbCk7XG4gICAgICAgIHF1ZXJ5ID0gYXBwbHlXaGVyZShxdWVyeSwgd2hlcmUpO1xuICAgICAgICBhd2FpdCBxdWVyeS5kZWwoKTtcbiAgICAgIH0sXG5cbiAgICAgIGRlbGV0ZU1hbnk6IGFzeW5jICh7IG1vZGVsLCB3aGVyZSB9KSA9PiB7XG4gICAgICAgIGxldCBxdWVyeSA9IGdldERiKCkobW9kZWwpO1xuICAgICAgICBxdWVyeSA9IGFwcGx5V2hlcmUocXVlcnksIHdoZXJlKTtcbiAgICAgICAgY29uc3QgY291bnQgPSBhd2FpdCBxdWVyeS5kZWwoKTtcbiAgICAgICAgcmV0dXJuIGNvdW50O1xuICAgICAgfSxcblxuICAgICAgY291bnQ6IGFzeW5jICh7IG1vZGVsLCB3aGVyZSB9KSA9PiB7XG4gICAgICAgIGxldCBxdWVyeSA9IGdldERiKCkobW9kZWwpO1xuICAgICAgICBpZiAod2hlcmUpIHtcbiAgICAgICAgICBxdWVyeSA9IGFwcGx5V2hlcmUocXVlcnksIHdoZXJlKTtcbiAgICAgICAgfVxuICAgICAgICBjb25zdCBbeyBjb3VudCB9XSA9IGF3YWl0IHF1ZXJ5LmNvdW50KFwiKiBhcyBjb3VudFwiKTtcbiAgICAgICAgcmV0dXJuIE51bWJlcihjb3VudCk7XG4gICAgICB9LFxuICAgIH0pO1xuICB9O1xuXG4gIGNvbnN0IGFkYXB0ZXJDb25maWcgPSB7XG4gICAgYWRhcHRlcklkOiBcInNvbmFtdS1rbmV4XCIsXG4gICAgYWRhcHRlck5hbWU6IFwiU29uYW11IEtuZXggQWRhcHRlclwiLFxuICAgIHVzZVBsdXJhbDogZmFsc2UsXG4gICAgc3VwcG9ydHNKU09OOiB0cnVlLFxuICAgIHN1cHBvcnRzRGF0ZXM6IHRydWUsXG4gICAgc3VwcG9ydHNCb29sZWFuczogdHJ1ZSxcbiAgICBzdXBwb3J0c051bWVyaWNJZHM6IGZhbHNlLFxuICAgIHRyYW5zYWN0aW9uOiBhc3luYyA8Uj4oY2I6ICh0cng6IERCVHJhbnNhY3Rpb25BZGFwdGVyKSA9PiBQcm9taXNlPFI+KTogUHJvbWlzZTxSPiA9PiB7XG4gICAgICBjb25zdCBkYiA9IERCLmdldERCKFwid1wiKTtcbiAgICAgIHJldHVybiBkYi50cmFuc2FjdGlvbihhc3luYyAodHJ4KSA9PiB7XG4gICAgICAgIGNvbnN0IG9wdGlvbnMgPSBsYXp5T3B0aW9ucztcbiAgICAgICAgaWYgKCFvcHRpb25zKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKFwic29uYW11S25leEFkYXB0ZXI6IG9wdGlvbnMgbm90IGluaXRpYWxpemVkXCIpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBjYihcbiAgICAgICAgICBjcmVhdGVBZGFwdGVyRmFjdG9yeSh7XG4gICAgICAgICAgICBjb25maWc6IGFkYXB0ZXJDb25maWcsXG4gICAgICAgICAgICBhZGFwdGVyOiBjcmVhdGVDdXN0b21BZGFwdGVyKCgpID0+IHRyeCksXG4gICAgICAgICAgfSkob3B0aW9ucyksXG4gICAgICAgICk7XG4gICAgICB9KTtcbiAgICB9LFxuICB9O1xuXG4gIGNvbnN0IGFkYXB0ZXJDcmVhdG9yID0gY3JlYXRlQWRhcHRlckZhY3Rvcnkoe1xuICAgIGNvbmZpZzogYWRhcHRlckNvbmZpZyxcbiAgICBhZGFwdGVyOiBjcmVhdGVDdXN0b21BZGFwdGVyKCgpID0+IERCLmdldERCKFwid1wiKSksXG4gIH0pO1xuXG4gIHJldHVybiAob3B0aW9uczogQmV0dGVyQXV0aE9wdGlvbnMpID0+IHtcbiAgICBsYXp5T3B0aW9ucyA9IG9wdGlvbnM7XG4gICAgcmV0dXJuIGFkYXB0ZXJDcmVhdG9yKG9wdGlvbnMpO1xuICB9O1xufTtcblxuLyoqXG4gKiBCZXR0ZXIgQXV0aOydmCDqs7Xsi50g7Ja064yR7YSwKEt5c2VseSwgRHJpenpsZSwgUHJpc21hLCBNb25nb0RCKSDtjKjthLTsl5Ag66ee7LawXG4gKiBBTkQg6re466O56rO8IE9SIOq3uOujueydhCDrtoTrpqztlZwg65KkIHRvcC1sZXZlbCBBTkTroZwg6rKw7ZWp7ZWp64uI64ukLlxuICog6rKw6rO8OiAoQSBBTkQgQiBBTkQgLi4uKSBBTkQgKEMgT1IgRCBPUiAuLi4pXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBhcHBseVdoZXJlKFxuICBxdWVyeTogS25leC5RdWVyeUJ1aWxkZXIsXG4gIGNvbmRpdGlvbnM6IENsZWFuZWRXaGVyZVtdLFxuKTogS25leC5RdWVyeUJ1aWxkZXIge1xuICBjb25zdCBhbmRHcm91cCA9IGNvbmRpdGlvbnMuZmlsdGVyKChjKSA9PiBjLmNvbm5lY3RvciAhPT0gXCJPUlwiKTtcbiAgY29uc3Qgb3JHcm91cCA9IGNvbmRpdGlvbnMuZmlsdGVyKChjKSA9PiBjLmNvbm5lY3RvciA9PT0gXCJPUlwiKTtcblxuICBpZiAoYW5kR3JvdXAubGVuZ3RoID4gMCkge1xuICAgIGZvciAoY29uc3QgY29uZGl0aW9uIG9mIGFuZEdyb3VwKSB7XG4gICAgICBxdWVyeSA9IGFwcGx5Q29uZGl0aW9uKHF1ZXJ5LCBjb25kaXRpb24sIFwid2hlcmVcIik7XG4gICAgfVxuICB9XG5cbiAgaWYgKG9yR3JvdXAubGVuZ3RoID4gMCkge1xuICAgIHF1ZXJ5ID0gcXVlcnkud2hlcmUoZnVuY3Rpb24gKHRoaXM6IEtuZXguUXVlcnlCdWlsZGVyKSB7XG4gICAgICBmb3IgKGxldCBpID0gMDsgaSA8IG9yR3JvdXAubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgYXBwbHlDb25kaXRpb24odGhpcywgb3JHcm91cFtpXSwgaSA9PT0gMCA/IFwid2hlcmVcIiA6IFwib3JXaGVyZVwiKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfVxuXG4gIHJldHVybiBxdWVyeTtcbn1cblxuZnVuY3Rpb24gYXBwbHlDb25kaXRpb24oXG4gIHF1ZXJ5OiBLbmV4LlF1ZXJ5QnVpbGRlcixcbiAgY29uZGl0aW9uOiBDbGVhbmVkV2hlcmUsXG4gIG1ldGhvZDogXCJ3aGVyZVwiIHwgXCJvcldoZXJlXCIsXG4pOiBLbmV4LlF1ZXJ5QnVpbGRlciB7XG4gIGNvbnN0IHsgZmllbGQsIHZhbHVlLCBvcGVyYXRvciB9ID0gY29uZGl0aW9uO1xuXG4gIHN3aXRjaCAob3BlcmF0b3IpIHtcbiAgICBjYXNlIFwiZXFcIjpcbiAgICAgIGlmICh2YWx1ZSA9PT0gbnVsbCkge1xuICAgICAgICByZXR1cm4gcXVlcnlbbWV0aG9kID09PSBcIm9yV2hlcmVcIiA/IFwib3JXaGVyZU51bGxcIiA6IFwid2hlcmVOdWxsXCJdKGZpZWxkKTtcbiAgICAgIH1cbiAgICAgIHJldHVybiBxdWVyeVttZXRob2RdKGZpZWxkLCBcIj1cIiwgdmFsdWUpO1xuICAgIGNhc2UgXCJuZVwiOlxuICAgICAgaWYgKHZhbHVlID09PSBudWxsKSB7XG4gICAgICAgIHJldHVybiBxdWVyeVttZXRob2QgPT09IFwib3JXaGVyZVwiID8gXCJvcldoZXJlTm90TnVsbFwiIDogXCJ3aGVyZU5vdE51bGxcIl0oZmllbGQpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIHF1ZXJ5W21ldGhvZF0oZmllbGQsIFwiIT1cIiwgdmFsdWUpO1xuICAgIGNhc2UgXCJsdFwiOlxuICAgICAgcmV0dXJuIHF1ZXJ5W21ldGhvZF0oZmllbGQsIFwiPFwiLCB2YWx1ZSk7XG4gICAgY2FzZSBcImx0ZVwiOlxuICAgICAgcmV0dXJuIHF1ZXJ5W21ldGhvZF0oZmllbGQsIFwiPD1cIiwgdmFsdWUpO1xuICAgIGNhc2UgXCJndFwiOlxuICAgICAgcmV0dXJuIHF1ZXJ5W21ldGhvZF0oZmllbGQsIFwiPlwiLCB2YWx1ZSk7XG4gICAgY2FzZSBcImd0ZVwiOlxuICAgICAgcmV0dXJuIHF1ZXJ5W21ldGhvZF0oZmllbGQsIFwiPj1cIiwgdmFsdWUpO1xuICAgIGNhc2UgXCJpblwiOlxuICAgICAgcmV0dXJuIHF1ZXJ5W21ldGhvZCA9PT0gXCJvcldoZXJlXCIgPyBcIm9yV2hlcmVJblwiIDogXCJ3aGVyZUluXCJdKFxuICAgICAgICBmaWVsZCxcbiAgICAgICAgdmFsdWUgYXMgKHN0cmluZyB8IG51bWJlcilbXSxcbiAgICAgICk7XG4gICAgY2FzZSBcIm5vdF9pblwiOlxuICAgICAgcmV0dXJuIHF1ZXJ5W21ldGhvZCA9PT0gXCJvcldoZXJlXCIgPyBcIm9yV2hlcmVOb3RJblwiIDogXCJ3aGVyZU5vdEluXCJdKFxuICAgICAgICBmaWVsZCxcbiAgICAgICAgdmFsdWUgYXMgKHN0cmluZyB8IG51bWJlcilbXSxcbiAgICAgICk7XG4gICAgY2FzZSBcImNvbnRhaW5zXCI6XG4gICAgICByZXR1cm4gcXVlcnlbbWV0aG9kXShmaWVsZCwgXCJsaWtlXCIsIGAlJHt2YWx1ZX0lYCk7XG4gICAgY2FzZSBcInN0YXJ0c193aXRoXCI6XG4gICAgICByZXR1cm4gcXVlcnlbbWV0aG9kXShmaWVsZCwgXCJsaWtlXCIsIGAke3ZhbHVlfSVgKTtcbiAgICBjYXNlIFwiZW5kc193aXRoXCI6XG4gICAgICByZXR1cm4gcXVlcnlbbWV0aG9kXShmaWVsZCwgXCJsaWtlXCIsIGAlJHt2YWx1ZX1gKTtcbiAgICBkZWZhdWx0OlxuICAgICAgcmV0dXJuIHF1ZXJ5O1xuICB9XG59XG4iXSwibmFtZXMiOlsiY3JlYXRlQWRhcHRlckZhY3RvcnkiLCJEQiIsInNvbmFtdUtuZXhBZGFwdGVyIiwibGF6eU9wdGlvbnMiLCJjcmVhdGVDdXN0b21BZGFwdGVyIiwiZ2V0RGIiLCJnZXRGaWVsZE5hbWUiLCJjcmVhdGUiLCJtb2RlbCIsImRhdGEiLCJyb3ciLCJpbnNlcnQiLCJyZXR1cm5pbmciLCJmaW5kT25lIiwid2hlcmUiLCJxdWVyeSIsImFwcGx5V2hlcmUiLCJmaXJzdCIsImZpbmRNYW55IiwibGltaXQiLCJvZmZzZXQiLCJzb3J0QnkiLCJkYkZpZWxkIiwiZmllbGQiLCJvcmRlckJ5IiwiZGlyZWN0aW9uIiwidXBkYXRlIiwidXBkYXRlTWFueSIsImNvdW50IiwiZGVsZXRlIiwiZGVsIiwiZGVsZXRlTWFueSIsIk51bWJlciIsImFkYXB0ZXJDb25maWciLCJhZGFwdGVySWQiLCJhZGFwdGVyTmFtZSIsInVzZVBsdXJhbCIsInN1cHBvcnRzSlNPTiIsInN1cHBvcnRzRGF0ZXMiLCJzdXBwb3J0c0Jvb2xlYW5zIiwic3VwcG9ydHNOdW1lcmljSWRzIiwidHJhbnNhY3Rpb24iLCJjYiIsImRiIiwiZ2V0REIiLCJ0cngiLCJvcHRpb25zIiwiRXJyb3IiLCJjb25maWciLCJhZGFwdGVyIiwiYWRhcHRlckNyZWF0b3IiLCJjb25kaXRpb25zIiwiYW5kR3JvdXAiLCJmaWx0ZXIiLCJjIiwiY29ubmVjdG9yIiwib3JHcm91cCIsImxlbmd0aCIsImNvbmRpdGlvbiIsImFwcGx5Q29uZGl0aW9uIiwiaSIsIm1ldGhvZCIsInZhbHVlIiwib3BlcmF0b3IiXSwibWFwcGluZ3MiOiJBQUtBLFNB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
|
|
@@ -723,13 +723,13 @@ export declare const admin: (options?: AdminOptions) => {
|
|
|
723
723
|
$Infer: {
|
|
724
724
|
body: ({
|
|
725
725
|
permission: {
|
|
726
|
-
readonly user?: ("update" | "delete" | "list" | "
|
|
726
|
+
readonly user?: ("update" | "delete" | "list" | "create" | "get" | "set-role" | "ban" | "impersonate" | "set-password")[] | undefined;
|
|
727
727
|
readonly session?: ("delete" | "list" | "revoke")[] | undefined;
|
|
728
728
|
};
|
|
729
729
|
permissions?: never | undefined;
|
|
730
730
|
} | {
|
|
731
731
|
permissions: {
|
|
732
|
-
readonly user?: ("update" | "delete" | "list" | "
|
|
732
|
+
readonly user?: ("update" | "delete" | "list" | "create" | "get" | "set-role" | "ban" | "impersonate" | "set-password")[] | undefined;
|
|
733
733
|
readonly session?: ("delete" | "list" | "revoke")[] | undefined;
|
|
734
734
|
};
|
|
735
735
|
permission?: never | undefined;
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
|
2
|
+
const tsLoaderRegisterStateKey = Symbol.for("sonamu.ts-loader-register.state");
|
|
3
|
+
function resetRegisterState() {
|
|
4
|
+
const globalState = globalThis;
|
|
5
|
+
delete globalState[tsLoaderRegisterStateKey];
|
|
6
|
+
delete process.env.SWCRC_PATH;
|
|
7
|
+
}
|
|
8
|
+
describe("ensureTsLoaderRegistered", ()=>{
|
|
9
|
+
beforeEach(()=>{
|
|
10
|
+
vi.resetModules();
|
|
11
|
+
vi.restoreAllMocks();
|
|
12
|
+
vi.unmock("node:module");
|
|
13
|
+
vi.unmock("../../utils/fs-utils.js");
|
|
14
|
+
resetRegisterState();
|
|
15
|
+
});
|
|
16
|
+
afterEach(()=>{
|
|
17
|
+
vi.resetModules();
|
|
18
|
+
vi.restoreAllMocks();
|
|
19
|
+
vi.unmock("node:module");
|
|
20
|
+
vi.unmock("../../utils/fs-utils.js");
|
|
21
|
+
resetRegisterState();
|
|
22
|
+
});
|
|
23
|
+
it("프로젝트 .swcrc를 우선 사용하고 중복 등록하지 않는다", async ()=>{
|
|
24
|
+
const registerMock = vi.fn();
|
|
25
|
+
vi.doMock("node:module", ()=>({
|
|
26
|
+
register: registerMock
|
|
27
|
+
}));
|
|
28
|
+
vi.doMock("../../utils/fs-utils.js", ()=>({
|
|
29
|
+
exists: vi.fn(async (candidate)=>candidate === "/tmp/fixture-api/.swcrc")
|
|
30
|
+
}));
|
|
31
|
+
const module = await import("../ts-loader-registration.js");
|
|
32
|
+
expect(registerMock).not.toHaveBeenCalled();
|
|
33
|
+
await module.ensureTsLoaderRegistered("/tmp/fixture-api");
|
|
34
|
+
expect(registerMock).toHaveBeenCalledTimes(1);
|
|
35
|
+
expect(registerMock).toHaveBeenCalledWith("@sonamu-kit/ts-loader/loader", {
|
|
36
|
+
parentURL: expect.stringContaining("/src/bin/ts-loader-registration")
|
|
37
|
+
});
|
|
38
|
+
expect(process.env.SWCRC_PATH).toBe("/tmp/fixture-api/.swcrc");
|
|
39
|
+
await module.ensureTsLoaderRegistered("/tmp/another-api");
|
|
40
|
+
expect(registerMock).toHaveBeenCalledTimes(1);
|
|
41
|
+
expect(process.env.SWCRC_PATH).toBe("/tmp/fixture-api/.swcrc");
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9iaW4vX190ZXN0c19fL3RzLWxvYWRlci1yZWdpc3Rlci50ZXN0LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGFmdGVyRWFjaCwgYmVmb3JlRWFjaCwgZGVzY3JpYmUsIGV4cGVjdCwgaXQsIHZpIH0gZnJvbSBcInZpdGVzdFwiO1xuXG5jb25zdCB0c0xvYWRlclJlZ2lzdGVyU3RhdGVLZXkgPSBTeW1ib2wuZm9yKFwic29uYW11LnRzLWxvYWRlci1yZWdpc3Rlci5zdGF0ZVwiKTtcblxudHlwZSBUc0xvYWRlclJlZ2lzdGVyU3RhdGUgPSB7XG4gIHJlZ2lzdGVyZWQ6IGJvb2xlYW47XG59O1xuXG50eXBlIEdsb2JhbFdpdGhUc0xvYWRlclJlZ2lzdGVyU3RhdGUgPSB0eXBlb2YgZ2xvYmFsVGhpcyAmIHtcbiAgW3RzTG9hZGVyUmVnaXN0ZXJTdGF0ZUtleV0/OiBUc0xvYWRlclJlZ2lzdGVyU3RhdGU7XG59O1xuXG5mdW5jdGlvbiByZXNldFJlZ2lzdGVyU3RhdGUoKSB7XG4gIGNvbnN0IGdsb2JhbFN0YXRlID0gZ2xvYmFsVGhpcyBhcyBHbG9iYWxXaXRoVHNMb2FkZXJSZWdpc3RlclN0YXRlO1xuICBkZWxldGUgZ2xvYmFsU3RhdGVbdHNMb2FkZXJSZWdpc3RlclN0YXRlS2V5XTtcbiAgZGVsZXRlIHByb2Nlc3MuZW52LlNXQ1JDX1BBVEg7XG59XG5cbmRlc2NyaWJlKFwiZW5zdXJlVHNMb2FkZXJSZWdpc3RlcmVkXCIsICgpID0+IHtcbiAgYmVmb3JlRWFjaCgoKSA9PiB7XG4gICAgdmkucmVzZXRNb2R1bGVzKCk7XG4gICAgdmkucmVzdG9yZUFsbE1vY2tzKCk7XG4gICAgdmkudW5tb2NrKFwibm9kZTptb2R1bGVcIik7XG4gICAgdmkudW5tb2NrKFwiLi4vLi4vdXRpbHMvZnMtdXRpbHMuanNcIik7XG4gICAgcmVzZXRSZWdpc3RlclN0YXRlKCk7XG4gIH0pO1xuXG4gIGFmdGVyRWFjaCgoKSA9PiB7XG4gICAgdmkucmVzZXRNb2R1bGVzKCk7XG4gICAgdmkucmVzdG9yZUFsbE1vY2tzKCk7XG4gICAgdmkudW5tb2NrKFwibm9kZTptb2R1bGVcIik7XG4gICAgdmkudW5tb2NrKFwiLi4vLi4vdXRpbHMvZnMtdXRpbHMuanNcIik7XG4gICAgcmVzZXRSZWdpc3RlclN0YXRlKCk7XG4gIH0pO1xuXG4gIGl0KFwi7ZSE66Gc7KCd7Yq4IC5zd2NyY+ulvCDsmrDshKAg7IKs7Jqp7ZWY6rOgIOykkeuztSDrk7HroZ3tlZjsp4Ag7JWK64qU64ukXCIsIGFzeW5jICgpID0+IHtcbiAgICBjb25zdCByZWdpc3Rlck1vY2sgPSB2aS5mbigpO1xuICAgIHZpLmRvTW9jayhcIm5vZGU6bW9kdWxlXCIsICgpID0+ICh7XG4gICAgICByZWdpc3RlcjogcmVnaXN0ZXJNb2NrLFxuICAgIH0pKTtcbiAgICB2aS5kb01vY2soXCIuLi8uLi91dGlscy9mcy11dGlscy5qc1wiLCAoKSA9PiAoe1xuICAgICAgZXhpc3RzOiB2aS5mbihhc3luYyAoY2FuZGlkYXRlOiBzdHJpbmcpID0+IGNhbmRpZGF0ZSA9PT0gXCIvdG1wL2ZpeHR1cmUtYXBpLy5zd2NyY1wiKSxcbiAgICB9KSk7XG5cbiAgICBjb25zdCBtb2R1bGUgPSBhd2FpdCBpbXBvcnQoXCIuLi90cy1sb2FkZXItcmVnaXN0cmF0aW9uXCIpO1xuXG4gICAgZXhwZWN0KHJlZ2lzdGVyTW9jaykubm90LnRvSGF2ZUJlZW5DYWxsZWQoKTtcblxuICAgIGF3YWl0IG1vZHVsZS5lbnN1cmVUc0xvYWRlclJlZ2lzdGVyZWQoXCIvdG1wL2ZpeHR1cmUtYXBpXCIpO1xuXG4gICAgZXhwZWN0KHJlZ2lzdGVyTW9jaykudG9IYXZlQmVlbkNhbGxlZFRpbWVzKDEpO1xuICAgIGV4cGVjdChyZWdpc3Rlck1vY2spLnRvSGF2ZUJlZW5DYWxsZWRXaXRoKFwiQHNvbmFtdS1raXQvdHMtbG9hZGVyL2xvYWRlclwiLCB7XG4gICAgICBwYXJlbnRVUkw6IGV4cGVjdC5zdHJpbmdDb250YWluaW5nKFwiL3NyYy9iaW4vdHMtbG9hZGVyLXJlZ2lzdHJhdGlvblwiKSxcbiAgICB9KTtcbiAgICBleHBlY3QocHJvY2Vzcy5lbnYuU1dDUkNfUEFUSCkudG9CZShcIi90bXAvZml4dHVyZS1hcGkvLnN3Y3JjXCIpO1xuXG4gICAgYXdhaXQgbW9kdWxlLmVuc3VyZVRzTG9hZGVyUmVnaXN0ZXJlZChcIi90bXAvYW5vdGhlci1hcGlcIik7XG5cbiAgICBleHBlY3QocmVnaXN0ZXJNb2NrKS50b0hhdmVCZWVuQ2FsbGVkVGltZXMoMSk7XG4gICAgZXhwZWN0KHByb2Nlc3MuZW52LlNXQ1JDX1BBVEgpLnRvQmUoXCIvdG1wL2ZpeHR1cmUtYXBpLy5zd2NyY1wiKTtcbiAgfSk7XG59KTtcbiJdLCJuYW1lcyI6WyJhZnRlckVhY2giLCJiZWZvcmVFYWNoIiwiZGVzY3JpYmUiLCJleHBlY3QiLCJpdCIsInZpIiwidHNMb2FkZXJSZWdpc3RlclN0YXRlS2V5IiwiU3ltYm9sIiwiZm9yIiwicmVzZXRSZWdpc3RlclN0YXRlIiwiZ2xvYmFsU3RhdGUiLCJnbG9iYWxUaGlzIiwicHJvY2VzcyIsImVudiIsIlNXQ1JDX1BBVEgiLCJyZXNldE1vZHVsZXMiLCJyZXN0b3JlQWxsTW9ja3MiLCJ1bm1vY2siLCJyZWdpc3Rlck1vY2siLCJmbiIsImRvTW9jayIsInJlZ2lzdGVyIiwiZXhpc3RzIiwiY2FuZGlkYXRlIiwibW9kdWxlIiwibm90IiwidG9IYXZlQmVlbkNhbGxlZCIsImVuc3VyZVRzTG9hZGVyUmVnaXN0ZXJlZCIsInRvSGF2ZUJlZW5DYWxsZWRUaW1lcyIsInRvSGF2ZUJlZW5DYWxsZWRXaXRoIiwicGFyZW50VVJMIiwic3RyaW5nQ29udGFpbmluZyIsInRvQmUiXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLFNBQVMsRUFBRUMsVUFBVSxFQUFFQyxRQUFRLEVBQUVDLE1BQU0sRUFBRUMsRUFBRSxFQUFFQyxFQUFFLFFBQVEsU0FBUztBQUV6RSxNQUFNQywyQkFBMkJDLE9BQU9DLEdBQUcsQ0FBQztBQVU1QyxTQUFTQztJQUNQLE1BQU1DLGNBQWNDO0lBQ3BCLE9BQU9ELFdBQVcsQ0FBQ0oseUJBQXlCO0lBQzVDLE9BQU9NLFFBQVFDLEdBQUcsQ0FBQ0MsVUFBVTtBQUMvQjtBQUVBWixTQUFTLDRCQUE0QjtJQUNuQ0QsV0FBVztRQUNUSSxHQUFHVSxZQUFZO1FBQ2ZWLEdBQUdXLGVBQWU7UUFDbEJYLEdBQUdZLE1BQU0sQ0FBQztRQUNWWixHQUFHWSxNQUFNLENBQUM7UUFDVlI7SUFDRjtJQUVBVCxVQUFVO1FBQ1JLLEdBQUdVLFlBQVk7UUFDZlYsR0FBR1csZUFBZTtRQUNsQlgsR0FBR1ksTUFBTSxDQUFDO1FBQ1ZaLEdBQUdZLE1BQU0sQ0FBQztRQUNWUjtJQUNGO0lBRUFMLEdBQUcsb0NBQW9DO1FBQ3JDLE1BQU1jLGVBQWViLEdBQUdjLEVBQUU7UUFDMUJkLEdBQUdlLE1BQU0sQ0FBQyxlQUFlLElBQU8sQ0FBQTtnQkFDOUJDLFVBQVVIO1lBQ1osQ0FBQTtRQUNBYixHQUFHZSxNQUFNLENBQUMsMkJBQTJCLElBQU8sQ0FBQTtnQkFDMUNFLFFBQVFqQixHQUFHYyxFQUFFLENBQUMsT0FBT0ksWUFBc0JBLGNBQWM7WUFDM0QsQ0FBQTtRQUVBLE1BQU1DLFNBQVMsTUFBTSxNQUFNLENBQUM7UUFFNUJyQixPQUFPZSxjQUFjTyxHQUFHLENBQUNDLGdCQUFnQjtRQUV6QyxNQUFNRixPQUFPRyx3QkFBd0IsQ0FBQztRQUV0Q3hCLE9BQU9lLGNBQWNVLHFCQUFxQixDQUFDO1FBQzNDekIsT0FBT2UsY0FBY1csb0JBQW9CLENBQUMsZ0NBQWdDO1lBQ3hFQyxXQUFXM0IsT0FBTzRCLGdCQUFnQixDQUFDO1FBQ3JDO1FBQ0E1QixPQUFPUyxRQUFRQyxHQUFHLENBQUNDLFVBQVUsRUFBRWtCLElBQUksQ0FBQztRQUVwQyxNQUFNUixPQUFPRyx3QkFBd0IsQ0FBQztRQUV0Q3hCLE9BQU9lLGNBQWNVLHFCQUFxQixDQUFDO1FBQzNDekIsT0FBT1MsUUFBUUMsR0FBRyxDQUFDQyxVQUFVLEVBQUVrQixJQUFJLENBQUM7SUFDdEM7QUFDRiJ9
|