medusa-product-helper 0.0.22 → 0.0.23
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,322 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GET = void 0;
|
|
4
|
+
const utils_1 = require("@medusajs/framework/utils");
|
|
5
|
+
const validators_1 = require("./validators");
|
|
6
|
+
/**
|
|
7
|
+
* GET /store/product-helper/inventory
|
|
8
|
+
*
|
|
9
|
+
* Returns product inventory levels (per variant) and pending order counts.
|
|
10
|
+
*
|
|
11
|
+
* Query parameters:
|
|
12
|
+
* - stock_only (boolean): If true, only return inventory data
|
|
13
|
+
* - order_only (boolean): If true, only return order counts
|
|
14
|
+
* - product_id (string[]): Filter by specific product IDs
|
|
15
|
+
* - limit (number): Pagination limit (default: 50)
|
|
16
|
+
* - offset (number): Pagination offset (default: 0)
|
|
17
|
+
*
|
|
18
|
+
* Note: Order counts are based on orders with status "pending".
|
|
19
|
+
*
|
|
20
|
+
* @example
|
|
21
|
+
* ```bash
|
|
22
|
+
* # Get both inventory and order counts
|
|
23
|
+
* GET /store/product-helper/inventory
|
|
24
|
+
*
|
|
25
|
+
* # Get only inventory
|
|
26
|
+
* GET /store/product-helper/inventory?stock_only=true
|
|
27
|
+
*
|
|
28
|
+
* # Get only order counts
|
|
29
|
+
* GET /store/product-helper/inventory?order_only=true
|
|
30
|
+
*
|
|
31
|
+
* # Filter by product IDs
|
|
32
|
+
* GET /store/product-helper/inventory?product_id=prod_123,prod_456
|
|
33
|
+
* ```
|
|
34
|
+
*/
|
|
35
|
+
const GET = async (req, res) => {
|
|
36
|
+
try {
|
|
37
|
+
// Validate query parameters
|
|
38
|
+
const validatedQuery = validators_1.InventoryQuerySchema.parse(req.query);
|
|
39
|
+
const { stock_only, order_only, product_id, limit, offset } = validatedQuery;
|
|
40
|
+
// Resolve query service
|
|
41
|
+
const query = req.scope.resolve(utils_1.ContainerRegistrationKeys.QUERY);
|
|
42
|
+
// Determine what data to fetch
|
|
43
|
+
const fetchInventory = !order_only || stock_only;
|
|
44
|
+
const fetchOrders = !stock_only || order_only;
|
|
45
|
+
// Build product filters
|
|
46
|
+
const productFilters = {};
|
|
47
|
+
if (product_id && product_id.length > 0) {
|
|
48
|
+
productFilters.id = product_id;
|
|
49
|
+
}
|
|
50
|
+
// Fetch products with variants
|
|
51
|
+
// Note: We fetch all products first (without pagination) to get accurate inventory/order data
|
|
52
|
+
// Then apply pagination to the final results
|
|
53
|
+
const { data: allProducts = [] } = await query.graph({
|
|
54
|
+
entity: "product",
|
|
55
|
+
fields: ["id", "title", "variants.id", "variants.title"],
|
|
56
|
+
filters: productFilters,
|
|
57
|
+
});
|
|
58
|
+
// If we only need orders, we'll filter products later based on which ones have orders
|
|
59
|
+
const products = allProducts;
|
|
60
|
+
// Build product map for quick lookup
|
|
61
|
+
// Initialize all products with ALL their variants (with 0 quantity initially)
|
|
62
|
+
// This ensures all variants are in the response, even if they don't have inventory items
|
|
63
|
+
const productMap = new Map();
|
|
64
|
+
for (const product of products) {
|
|
65
|
+
const productData = product;
|
|
66
|
+
// Initialize variants array with all variants from the product (with 0 quantity)
|
|
67
|
+
const variants = [];
|
|
68
|
+
if (productData.variants && Array.isArray(productData.variants)) {
|
|
69
|
+
for (const variant of productData.variants) {
|
|
70
|
+
if (variant.id) {
|
|
71
|
+
variants.push({
|
|
72
|
+
variant_id: variant.id,
|
|
73
|
+
variant_title: variant.title || "Default Variant",
|
|
74
|
+
inventory_quantity: 0, // Initialize with 0, will be updated if inventory exists
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
// Calculate total inventory quantity (sum of all variants)
|
|
80
|
+
const totalInventoryQuantity = variants.reduce((sum, variant) => sum + variant.inventory_quantity, 0);
|
|
81
|
+
productMap.set(productData.id, {
|
|
82
|
+
product_id: productData.id,
|
|
83
|
+
product_title: productData.title || "Unknown Product",
|
|
84
|
+
variants, // All variants initialized with 0 quantity
|
|
85
|
+
total_inventory_quantity: totalInventoryQuantity, // Sum of all variant inventory quantities
|
|
86
|
+
confirmed_order_count: 0, // Initialize with 0, will be updated if orders exist
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
// Fetch inventory data if needed
|
|
90
|
+
if (fetchInventory) {
|
|
91
|
+
// Query inventory items with variant relationships
|
|
92
|
+
const { data: inventoryItems = [] } = await query.graph({
|
|
93
|
+
entity: "inventory_item",
|
|
94
|
+
fields: [
|
|
95
|
+
"id",
|
|
96
|
+
"variants.id",
|
|
97
|
+
],
|
|
98
|
+
});
|
|
99
|
+
// Create map of variant_id -> inventory_item_id
|
|
100
|
+
const variantToInventoryItem = new Map();
|
|
101
|
+
for (const inventoryItem of inventoryItems) {
|
|
102
|
+
const item = inventoryItem;
|
|
103
|
+
if (item.variants && Array.isArray(item.variants)) {
|
|
104
|
+
for (const variant of item.variants) {
|
|
105
|
+
if (variant.id) {
|
|
106
|
+
variantToInventoryItem.set(variant.id, item.id);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
// Query location levels for all inventory items
|
|
112
|
+
const inventoryItemIds = Array.from(new Set(variantToInventoryItem.values()));
|
|
113
|
+
if (inventoryItemIds.length > 0) {
|
|
114
|
+
// Try to get location levels via inventory service first
|
|
115
|
+
try {
|
|
116
|
+
const { Modules } = await import("@medusajs/framework/utils");
|
|
117
|
+
const inventoryService = req.scope.resolve(Modules.INVENTORY);
|
|
118
|
+
if (inventoryService?.listInventoryLevels) {
|
|
119
|
+
const locationLevels = await inventoryService.listInventoryLevels({
|
|
120
|
+
inventory_item_id: inventoryItemIds,
|
|
121
|
+
});
|
|
122
|
+
// Aggregate inventory per inventory item
|
|
123
|
+
const inventoryItemQuantityMap = new Map();
|
|
124
|
+
for (const level of locationLevels || []) {
|
|
125
|
+
const current = inventoryItemQuantityMap.get(level.inventory_item_id) || 0;
|
|
126
|
+
inventoryItemQuantityMap.set(level.inventory_item_id, current + (level.available_quantity || 0));
|
|
127
|
+
}
|
|
128
|
+
// Map inventory quantities to variants
|
|
129
|
+
// Update existing variants with actual inventory quantities
|
|
130
|
+
for (const [variantId, inventoryItemId] of variantToInventoryItem) {
|
|
131
|
+
const quantity = inventoryItemQuantityMap.get(inventoryItemId) || 0;
|
|
132
|
+
// Find which product this variant belongs to and update the variant
|
|
133
|
+
for (const product of products) {
|
|
134
|
+
const productData = product;
|
|
135
|
+
if (productData.variants) {
|
|
136
|
+
const variant = productData.variants.find((v) => v.id === variantId);
|
|
137
|
+
if (variant) {
|
|
138
|
+
const productEntry = productMap.get(productData.id);
|
|
139
|
+
if (productEntry && productEntry.variants) {
|
|
140
|
+
// Find and update the existing variant instead of adding a new one
|
|
141
|
+
const existingVariant = productEntry.variants.find((v) => v.variant_id === variantId);
|
|
142
|
+
if (existingVariant) {
|
|
143
|
+
existingVariant.inventory_quantity = quantity;
|
|
144
|
+
// Recalculate total inventory quantity for this product
|
|
145
|
+
productEntry.total_inventory_quantity = productEntry.variants.reduce((sum, v) => sum + v.inventory_quantity, 0);
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
break;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
catch (error) {
|
|
156
|
+
// Fallback: Query location levels via graph query
|
|
157
|
+
const { data: itemsWithLevels = [] } = await query.graph({
|
|
158
|
+
entity: "inventory_item",
|
|
159
|
+
fields: [
|
|
160
|
+
"id",
|
|
161
|
+
"location_levels.available_quantity",
|
|
162
|
+
],
|
|
163
|
+
filters: { id: inventoryItemIds },
|
|
164
|
+
});
|
|
165
|
+
// Aggregate inventory per inventory item
|
|
166
|
+
const inventoryItemQuantityMap = new Map();
|
|
167
|
+
for (const item of itemsWithLevels) {
|
|
168
|
+
const itemData = item;
|
|
169
|
+
if (itemData.location_levels && Array.isArray(itemData.location_levels)) {
|
|
170
|
+
const totalQuantity = itemData.location_levels.reduce((sum, level) => sum + (level.available_quantity || 0), 0);
|
|
171
|
+
inventoryItemQuantityMap.set(itemData.id, totalQuantity);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
// Map inventory quantities to variants
|
|
175
|
+
// Update existing variants with actual inventory quantities
|
|
176
|
+
for (const [variantId, inventoryItemId] of variantToInventoryItem) {
|
|
177
|
+
const quantity = inventoryItemQuantityMap.get(inventoryItemId) || 0;
|
|
178
|
+
// Find which product this variant belongs to and update the variant
|
|
179
|
+
for (const product of products) {
|
|
180
|
+
const productData = product;
|
|
181
|
+
if (productData.variants) {
|
|
182
|
+
const variant = productData.variants.find((v) => v.id === variantId);
|
|
183
|
+
if (variant) {
|
|
184
|
+
const productEntry = productMap.get(productData.id);
|
|
185
|
+
if (productEntry && productEntry.variants) {
|
|
186
|
+
// Find and update the existing variant instead of adding a new one
|
|
187
|
+
const existingVariant = productEntry.variants.find((v) => v.variant_id === variantId);
|
|
188
|
+
if (existingVariant) {
|
|
189
|
+
existingVariant.inventory_quantity = quantity;
|
|
190
|
+
// Recalculate total inventory quantity for this product
|
|
191
|
+
productEntry.total_inventory_quantity = productEntry.variants.reduce((sum, v) => sum + v.inventory_quantity, 0);
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
break;
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
// Fetch pending order counts if needed
|
|
203
|
+
if (fetchOrders) {
|
|
204
|
+
// Query pending orders with items (using OrderStatus enum)
|
|
205
|
+
const { data: orders = [] } = await query.graph({
|
|
206
|
+
entity: "order",
|
|
207
|
+
fields: ["id", "items.variant_id"],
|
|
208
|
+
filters: { status: utils_1.OrderStatus.PENDING },
|
|
209
|
+
});
|
|
210
|
+
// Get all variant IDs from order items
|
|
211
|
+
const variantIds = new Set();
|
|
212
|
+
for (const order of orders) {
|
|
213
|
+
const orderData = order;
|
|
214
|
+
if (orderData.items && Array.isArray(orderData.items)) {
|
|
215
|
+
for (const item of orderData.items) {
|
|
216
|
+
if (item.variant_id) {
|
|
217
|
+
variantIds.add(item.variant_id);
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
// Query variants to get product_id mapping
|
|
223
|
+
const variantIdsArray = Array.from(variantIds);
|
|
224
|
+
const variantToProductMap = new Map();
|
|
225
|
+
if (variantIdsArray.length > 0) {
|
|
226
|
+
const { data: variants = [] } = await query.graph({
|
|
227
|
+
entity: "product_variant",
|
|
228
|
+
fields: ["id", "product_id"],
|
|
229
|
+
filters: { id: variantIdsArray },
|
|
230
|
+
});
|
|
231
|
+
for (const variant of variants) {
|
|
232
|
+
const variantData = variant;
|
|
233
|
+
if (variantData.product_id) {
|
|
234
|
+
variantToProductMap.set(variantData.id, variantData.product_id);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
// Count pending orders per product
|
|
239
|
+
const productOrderCountMap = new Map();
|
|
240
|
+
for (const order of orders) {
|
|
241
|
+
const orderData = order;
|
|
242
|
+
if (orderData.items && Array.isArray(orderData.items)) {
|
|
243
|
+
const productIdsInOrder = new Set();
|
|
244
|
+
for (const item of orderData.items) {
|
|
245
|
+
if (item.variant_id) {
|
|
246
|
+
const productId = variantToProductMap.get(item.variant_id);
|
|
247
|
+
if (productId) {
|
|
248
|
+
productIdsInOrder.add(productId);
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
// Count this order once per unique product
|
|
253
|
+
for (const productId of productIdsInOrder) {
|
|
254
|
+
const current = productOrderCountMap.get(productId) || 0;
|
|
255
|
+
productOrderCountMap.set(productId, current + 1);
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
// Add pending order counts to product map
|
|
260
|
+
for (const [productId, count] of productOrderCountMap) {
|
|
261
|
+
const productEntry = productMap.get(productId);
|
|
262
|
+
if (productEntry) {
|
|
263
|
+
productEntry.confirmed_order_count = count;
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
// Recalculate total inventory quantity for all products (in case any were missed)
|
|
268
|
+
for (const productEntry of productMap.values()) {
|
|
269
|
+
if (productEntry.variants && productEntry.variants.length > 0) {
|
|
270
|
+
productEntry.total_inventory_quantity = productEntry.variants.reduce((sum, variant) => sum + variant.inventory_quantity, 0);
|
|
271
|
+
}
|
|
272
|
+
else {
|
|
273
|
+
productEntry.total_inventory_quantity = 0;
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
// Convert map to array and filter based on query params
|
|
277
|
+
let resultProducts = Array.from(productMap.values());
|
|
278
|
+
// Note: We don't filter out products without variants anymore
|
|
279
|
+
// Products without variants will still show with empty variants array
|
|
280
|
+
// and will show order counts if they have any
|
|
281
|
+
// If order_only, only show products that have order counts
|
|
282
|
+
if (order_only && !stock_only) {
|
|
283
|
+
resultProducts = resultProducts.filter((p) => p.confirmed_order_count !== undefined && p.confirmed_order_count > 0);
|
|
284
|
+
}
|
|
285
|
+
// Get total count before pagination (for pagination metadata)
|
|
286
|
+
const totalCount = resultProducts.length;
|
|
287
|
+
// Apply pagination to final results
|
|
288
|
+
const paginatedProducts = resultProducts.slice(offset, offset + limit);
|
|
289
|
+
const response = {
|
|
290
|
+
products: paginatedProducts,
|
|
291
|
+
count: totalCount,
|
|
292
|
+
offset,
|
|
293
|
+
limit,
|
|
294
|
+
};
|
|
295
|
+
res.json(response);
|
|
296
|
+
}
|
|
297
|
+
catch (error) {
|
|
298
|
+
// Handle validation errors
|
|
299
|
+
if (error && typeof error === "object" && "issues" in error) {
|
|
300
|
+
res.status(400).json({
|
|
301
|
+
message: "Invalid query parameters",
|
|
302
|
+
errors: error.issues,
|
|
303
|
+
products: [],
|
|
304
|
+
count: 0,
|
|
305
|
+
offset: 0,
|
|
306
|
+
limit: 50,
|
|
307
|
+
});
|
|
308
|
+
return;
|
|
309
|
+
}
|
|
310
|
+
// Handle other errors
|
|
311
|
+
const errorMessage = error instanceof Error ? error.message : "Internal server error";
|
|
312
|
+
res.status(500).json({
|
|
313
|
+
message: errorMessage,
|
|
314
|
+
products: [],
|
|
315
|
+
count: 0,
|
|
316
|
+
offset: 0,
|
|
317
|
+
limit: 50,
|
|
318
|
+
});
|
|
319
|
+
}
|
|
320
|
+
};
|
|
321
|
+
exports.GET = GET;
|
|
322
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"route.js","sourceRoot":"","sources":["../../../../../../../src/api/store/product-helper/inventory/route.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.InventoryQuerySchema = void 0;
|
|
4
|
+
const zod_1 = require("zod");
|
|
5
|
+
const booleanish = zod_1.z
|
|
6
|
+
.union([zod_1.z.boolean(), zod_1.z.string(), zod_1.z.number()])
|
|
7
|
+
.optional()
|
|
8
|
+
.transform((value) => {
|
|
9
|
+
if (typeof value === "boolean") {
|
|
10
|
+
return value;
|
|
11
|
+
}
|
|
12
|
+
if (typeof value === "number") {
|
|
13
|
+
return value !== 0;
|
|
14
|
+
}
|
|
15
|
+
if (typeof value === "string") {
|
|
16
|
+
const normalized = value.trim().toLowerCase();
|
|
17
|
+
if (["true", "1", "yes", "y"].includes(normalized)) {
|
|
18
|
+
return true;
|
|
19
|
+
}
|
|
20
|
+
if (["false", "0", "no", "n"].includes(normalized)) {
|
|
21
|
+
return false;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
return undefined;
|
|
25
|
+
});
|
|
26
|
+
const positiveInt = zod_1.z
|
|
27
|
+
.union([zod_1.z.string(), zod_1.z.number()])
|
|
28
|
+
.default(50)
|
|
29
|
+
.transform((value) => {
|
|
30
|
+
const parsed = Number(value);
|
|
31
|
+
if (Number.isNaN(parsed) || parsed <= 0) {
|
|
32
|
+
return 50;
|
|
33
|
+
}
|
|
34
|
+
return Math.floor(parsed);
|
|
35
|
+
});
|
|
36
|
+
const nonNegativeInt = zod_1.z
|
|
37
|
+
.union([zod_1.z.string(), zod_1.z.number()])
|
|
38
|
+
.default(0)
|
|
39
|
+
.transform((value) => {
|
|
40
|
+
const parsed = Number(value);
|
|
41
|
+
if (Number.isNaN(parsed) || parsed < 0) {
|
|
42
|
+
return 0;
|
|
43
|
+
}
|
|
44
|
+
return Math.floor(parsed);
|
|
45
|
+
});
|
|
46
|
+
const stringArray = zod_1.z
|
|
47
|
+
.union([
|
|
48
|
+
zod_1.z.string().transform((value) => value
|
|
49
|
+
.split(",")
|
|
50
|
+
.map((entry) => entry.trim())
|
|
51
|
+
.filter(Boolean)),
|
|
52
|
+
zod_1.z.array(zod_1.z.string()),
|
|
53
|
+
])
|
|
54
|
+
.optional()
|
|
55
|
+
.transform((value) => (Array.isArray(value) ? value : []));
|
|
56
|
+
exports.InventoryQuerySchema = zod_1.z.object({
|
|
57
|
+
stock_only: booleanish,
|
|
58
|
+
order_only: booleanish,
|
|
59
|
+
product_id: stringArray,
|
|
60
|
+
limit: positiveInt,
|
|
61
|
+
offset: nonNegativeInt,
|
|
62
|
+
});
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmFsaWRhdG9ycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3NyYy9hcGkvc3RvcmUvcHJvZHVjdC1oZWxwZXIvaW52ZW50b3J5L3ZhbGlkYXRvcnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkJBQXVCO0FBRXZCLE1BQU0sVUFBVSxHQUFHLE9BQUM7S0FDakIsS0FBSyxDQUFDLENBQUMsT0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFLE9BQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRSxPQUFDLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztLQUM1QyxRQUFRLEVBQUU7S0FDVixTQUFTLENBQUMsQ0FBQyxLQUE0QyxFQUFFLEVBQUU7SUFDMUQsSUFBSSxPQUFPLEtBQUssS0FBSyxTQUFTLEVBQUUsQ0FBQztRQUMvQixPQUFPLEtBQUssQ0FBQTtJQUNkLENBQUM7SUFFRCxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQzlCLE9BQU8sS0FBSyxLQUFLLENBQUMsQ0FBQTtJQUNwQixDQUFDO0lBRUQsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztRQUM5QixNQUFNLFVBQVUsR0FBRyxLQUFLLENBQUMsSUFBSSxFQUFFLENBQUMsV0FBVyxFQUFFLENBQUE7UUFDN0MsSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLEVBQUUsS0FBSyxFQUFFLEdBQUcsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxVQUFVLENBQUMsRUFBRSxDQUFDO1lBQ25ELE9BQU8sSUFBSSxDQUFBO1FBQ2IsQ0FBQztRQUNELElBQUksQ0FBQyxPQUFPLEVBQUUsR0FBRyxFQUFFLElBQUksRUFBRSxHQUFHLENBQUMsQ0FBQyxRQUFRLENBQUMsVUFBVSxDQUFDLEVBQUUsQ0FBQztZQUNuRCxPQUFPLEtBQUssQ0FBQTtRQUNkLENBQUM7SUFDSCxDQUFDO0lBRUQsT0FBTyxTQUFTLENBQUE7QUFDbEIsQ0FBQyxDQUFDLENBQUE7QUFFSixNQUFNLFdBQVcsR0FBRyxPQUFDO0tBQ2xCLEtBQUssQ0FBQyxDQUFDLE9BQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRSxPQUFDLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztLQUMvQixPQUFPLENBQUMsRUFBRSxDQUFDO0tBQ1gsU0FBUyxDQUFDLENBQUMsS0FBc0IsRUFBRSxFQUFFO0lBQ3BDLE1BQU0sTUFBTSxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQTtJQUM1QixJQUFJLE1BQU0sQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLElBQUksTUFBTSxJQUFJLENBQUMsRUFBRSxDQUFDO1FBQ3hDLE9BQU8sRUFBRSxDQUFBO0lBQ1gsQ0FBQztJQUNELE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsQ0FBQTtBQUMzQixDQUFDLENBQUMsQ0FBQTtBQUVKLE1BQU0sY0FBYyxHQUFHLE9BQUM7S0FDckIsS0FBSyxDQUFDLENBQUMsT0FBQyxDQUFDLE1BQU0sRUFBRSxFQUFFLE9BQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDO0tBQy9CLE9BQU8sQ0FBQyxDQUFDLENBQUM7S0FDVixTQUFTLENBQUMsQ0FBQyxLQUFzQixFQUFFLEVBQUU7SUFDcEMsTUFBTSxNQUFNLEdBQUcsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFBO0lBQzVCLElBQUksTUFBTSxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsSUFBSSxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7UUFDdkMsT0FBTyxDQUFDLENBQUE7SUFDVixDQUFDO0lBQ0QsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFBO0FBQzNCLENBQUMsQ0FBQyxDQUFBO0FBRUosTUFBTSxXQUFXLEdBQUcsT0FBQztLQUNsQixLQUFLLENBQUM7SUFDTCxPQUFDLENBQUMsTUFBTSxFQUFFLENBQUMsU0FBUyxDQUFDLENBQUMsS0FBYSxFQUFFLEVBQUUsQ0FDckMsS0FBSztTQUNGLEtBQUssQ0FBQyxHQUFHLENBQUM7U0FDVixHQUFHLENBQUMsQ0FBQyxLQUFhLEVBQUUsRUFBRSxDQUFDLEtBQUssQ0FBQyxJQUFJLEVBQUUsQ0FBQztTQUNwQyxNQUFNLENBQUMsT0FBTyxDQUFDLENBQ25CO0lBQ0QsT0FBQyxDQUFDLEtBQUssQ0FBQyxPQUFDLENBQUMsTUFBTSxFQUFFLENBQUM7Q0FDcEIsQ0FBQztLQUNELFFBQVEsRUFBRTtLQUNWLFNBQVMsQ0FBQyxDQUFDLEtBQW9DLEVBQUUsRUFBRSxDQUFDLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFBO0FBRTlFLFFBQUEsb0JBQW9CLEdBQUcsT0FBQyxDQUFDLE1BQU0sQ0FBQztJQUMzQyxVQUFVLEVBQUUsVUFBVTtJQUN0QixVQUFVLEVBQUUsVUFBVTtJQUN0QixVQUFVLEVBQUUsV0FBVztJQUN2QixLQUFLLEVBQUUsV0FBVztJQUNsQixNQUFNLEVBQUUsY0FBYztDQUN2QixDQUFDLENBQUEifQ==
|