@lodashventure/medusa-sales-analytics 1.0.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.
- package/.medusa/server/api/admin/analytics/sales/route.d.ts +2 -0
- package/.medusa/server/api/admin/analytics/sales/route.js +177 -0
- package/.medusa/server/api/admin/products/[id]/sales-history/route.d.ts +2 -0
- package/.medusa/server/api/admin/products/[id]/sales-history/route.js +204 -0
- package/.medusa/server/api/admin/products/sales-statistics/route.d.ts +2 -0
- package/.medusa/server/api/admin/products/sales-statistics/route.js +97 -0
- package/.medusa/server/api/admin/sales/summary/route.d.ts +2 -0
- package/.medusa/server/api/admin/sales/summary/route.js +176 -0
- package/.medusa/server/api/middlewares/add-product-filters.d.ts +7 -0
- package/.medusa/server/api/middlewares/add-product-filters.js +173 -0
- package/.medusa/server/api/middlewares/add-sales-statistics.d.ts +2 -0
- package/.medusa/server/api/middlewares/add-sales-statistics.js +103 -0
- package/.medusa/server/api/middlewares/add-stock-availability.d.ts +7 -0
- package/.medusa/server/api/middlewares/add-stock-availability.js +91 -0
- package/.medusa/server/api/middlewares.d.ts +2 -0
- package/.medusa/server/api/middlewares.js +36 -0
- package/.medusa/server/index.d.ts +2 -0
- package/.medusa/server/index.js +8 -0
- package/README.md +60 -0
- package/package.json +48 -0
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GET = GET;
|
|
4
|
+
const utils_1 = require("@medusajs/framework/utils");
|
|
5
|
+
async function GET(req, res) {
|
|
6
|
+
const query = req.scope.resolve(utils_1.ContainerRegistrationKeys.QUERY);
|
|
7
|
+
// Parse query parameters
|
|
8
|
+
const startDate = req.query.start_date ? new Date(req.query.start_date) : new Date(new Date().setMonth(new Date().getMonth() - 1));
|
|
9
|
+
const endDate = req.query.end_date ? new Date(req.query.end_date) : new Date();
|
|
10
|
+
const groupBy = req.query.group_by || "day"; // day, week, month, year
|
|
11
|
+
const productId = req.query.product_id;
|
|
12
|
+
const collectionId = req.query.collection_id;
|
|
13
|
+
const categoryId = req.query.category_id;
|
|
14
|
+
// Build filters
|
|
15
|
+
let filters = {};
|
|
16
|
+
if (productId)
|
|
17
|
+
filters.product_id = productId;
|
|
18
|
+
if (collectionId)
|
|
19
|
+
filters.collection_id = collectionId;
|
|
20
|
+
if (categoryId)
|
|
21
|
+
filters.category_id = categoryId;
|
|
22
|
+
// Get products based on filters
|
|
23
|
+
const { data: products } = await query.graph({
|
|
24
|
+
entity: "product",
|
|
25
|
+
fields: ["id", "title", "variants.*", "collection.*", "categories.*"],
|
|
26
|
+
filters: filters
|
|
27
|
+
});
|
|
28
|
+
// Get all order items in date range
|
|
29
|
+
const { data: orderItems } = await query.graph({
|
|
30
|
+
entity: "order_line_item",
|
|
31
|
+
fields: [
|
|
32
|
+
"id",
|
|
33
|
+
"variant_id",
|
|
34
|
+
"product_id",
|
|
35
|
+
"quantity",
|
|
36
|
+
"unit_price",
|
|
37
|
+
"subtotal",
|
|
38
|
+
"order.*"
|
|
39
|
+
]
|
|
40
|
+
});
|
|
41
|
+
// Filter for completed orders and date range
|
|
42
|
+
const completedOrderItems = orderItems.filter(item => {
|
|
43
|
+
if (!item.order)
|
|
44
|
+
return false;
|
|
45
|
+
const orderDate = new Date(item.order.created_at);
|
|
46
|
+
return orderDate >= startDate && orderDate <= endDate && [
|
|
47
|
+
"completed",
|
|
48
|
+
"partially_returned",
|
|
49
|
+
"partially_shipped",
|
|
50
|
+
"shipped",
|
|
51
|
+
"fulfilled",
|
|
52
|
+
"partially_fulfilled"
|
|
53
|
+
].includes(item.order.status);
|
|
54
|
+
});
|
|
55
|
+
// Group sales by period
|
|
56
|
+
const salesByPeriod = {};
|
|
57
|
+
completedOrderItems.forEach(item => {
|
|
58
|
+
const orderDate = new Date(item.order.created_at);
|
|
59
|
+
let periodKey;
|
|
60
|
+
switch (groupBy) {
|
|
61
|
+
case "day":
|
|
62
|
+
periodKey = orderDate.toISOString().split("T")[0];
|
|
63
|
+
break;
|
|
64
|
+
case "week":
|
|
65
|
+
const weekStart = new Date(orderDate);
|
|
66
|
+
weekStart.setDate(orderDate.getDate() - orderDate.getDay());
|
|
67
|
+
periodKey = weekStart.toISOString().split("T")[0];
|
|
68
|
+
break;
|
|
69
|
+
case "month":
|
|
70
|
+
periodKey = `${orderDate.getFullYear()}-${String(orderDate.getMonth() + 1).padStart(2, '0')}`;
|
|
71
|
+
break;
|
|
72
|
+
case "year":
|
|
73
|
+
periodKey = orderDate.getFullYear().toString();
|
|
74
|
+
break;
|
|
75
|
+
default:
|
|
76
|
+
periodKey = orderDate.toISOString().split("T")[0];
|
|
77
|
+
}
|
|
78
|
+
if (!salesByPeriod[periodKey]) {
|
|
79
|
+
salesByPeriod[periodKey] = {
|
|
80
|
+
date: periodKey,
|
|
81
|
+
revenue: 0,
|
|
82
|
+
quantity: 0,
|
|
83
|
+
orders: new Set(),
|
|
84
|
+
products_sold: new Set()
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
salesByPeriod[periodKey].revenue += item.subtotal || 0;
|
|
88
|
+
salesByPeriod[periodKey].quantity += item.quantity || 0;
|
|
89
|
+
salesByPeriod[periodKey].orders.add(item.order.id);
|
|
90
|
+
salesByPeriod[periodKey].products_sold.add(item.product_id);
|
|
91
|
+
});
|
|
92
|
+
// Convert to array and calculate final metrics
|
|
93
|
+
const chartData = Object.keys(salesByPeriod)
|
|
94
|
+
.sort()
|
|
95
|
+
.map(key => ({
|
|
96
|
+
date: key,
|
|
97
|
+
revenue: salesByPeriod[key].revenue,
|
|
98
|
+
quantity: salesByPeriod[key].quantity,
|
|
99
|
+
order_count: salesByPeriod[key].orders.size,
|
|
100
|
+
unique_products: salesByPeriod[key].products_sold.size,
|
|
101
|
+
average_order_value: salesByPeriod[key].orders.size > 0
|
|
102
|
+
? salesByPeriod[key].revenue / salesByPeriod[key].orders.size
|
|
103
|
+
: 0
|
|
104
|
+
}));
|
|
105
|
+
// Calculate summary statistics
|
|
106
|
+
const totalRevenue = completedOrderItems.reduce((sum, item) => sum + (item.subtotal || 0), 0);
|
|
107
|
+
const totalQuantity = completedOrderItems.reduce((sum, item) => sum + (item.quantity || 0), 0);
|
|
108
|
+
const uniqueOrders = new Set(completedOrderItems.map(item => item.order?.id).filter(Boolean));
|
|
109
|
+
const uniqueCustomers = new Set(completedOrderItems.map(item => item.order?.customer_id).filter(Boolean));
|
|
110
|
+
// Top products
|
|
111
|
+
const productSales = {};
|
|
112
|
+
completedOrderItems.forEach(item => {
|
|
113
|
+
const product = products.find(p => p.variants?.some((v) => v.id === item.variant_id));
|
|
114
|
+
if (product) {
|
|
115
|
+
if (!productSales[product.id]) {
|
|
116
|
+
productSales[product.id] = {
|
|
117
|
+
product_id: product.id,
|
|
118
|
+
product_title: product.title,
|
|
119
|
+
quantity_sold: 0,
|
|
120
|
+
revenue: 0
|
|
121
|
+
};
|
|
122
|
+
}
|
|
123
|
+
productSales[product.id].quantity_sold += item.quantity || 0;
|
|
124
|
+
productSales[product.id].revenue += item.subtotal || 0;
|
|
125
|
+
}
|
|
126
|
+
});
|
|
127
|
+
const topProducts = Object.values(productSales)
|
|
128
|
+
.sort((a, b) => b.revenue - a.revenue)
|
|
129
|
+
.slice(0, 10);
|
|
130
|
+
res.json({
|
|
131
|
+
analytics: {
|
|
132
|
+
period: {
|
|
133
|
+
start_date: startDate,
|
|
134
|
+
end_date: endDate,
|
|
135
|
+
group_by: groupBy
|
|
136
|
+
},
|
|
137
|
+
summary: {
|
|
138
|
+
total_revenue: totalRevenue,
|
|
139
|
+
total_quantity_sold: totalQuantity,
|
|
140
|
+
total_orders: uniqueOrders.size,
|
|
141
|
+
unique_customers: uniqueCustomers.size,
|
|
142
|
+
average_order_value: uniqueOrders.size > 0 ? totalRevenue / uniqueOrders.size : 0,
|
|
143
|
+
average_items_per_order: uniqueOrders.size > 0 ? totalQuantity / uniqueOrders.size : 0
|
|
144
|
+
},
|
|
145
|
+
chart_data: chartData,
|
|
146
|
+
top_products: topProducts,
|
|
147
|
+
growth: calculateGrowth(chartData)
|
|
148
|
+
}
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
function calculateGrowth(data) {
|
|
152
|
+
if (data.length < 2)
|
|
153
|
+
return null;
|
|
154
|
+
const current = data[data.length - 1];
|
|
155
|
+
const previous = data[data.length - 2];
|
|
156
|
+
return {
|
|
157
|
+
revenue: {
|
|
158
|
+
value: current.revenue - previous.revenue,
|
|
159
|
+
percentage: previous.revenue > 0
|
|
160
|
+
? ((current.revenue - previous.revenue) / previous.revenue * 100).toFixed(2)
|
|
161
|
+
: 0
|
|
162
|
+
},
|
|
163
|
+
quantity: {
|
|
164
|
+
value: current.quantity - previous.quantity,
|
|
165
|
+
percentage: previous.quantity > 0
|
|
166
|
+
? ((current.quantity - previous.quantity) / previous.quantity * 100).toFixed(2)
|
|
167
|
+
: 0
|
|
168
|
+
},
|
|
169
|
+
orders: {
|
|
170
|
+
value: current.order_count - previous.order_count,
|
|
171
|
+
percentage: previous.order_count > 0
|
|
172
|
+
? ((current.order_count - previous.order_count) / previous.order_count * 100).toFixed(2)
|
|
173
|
+
: 0
|
|
174
|
+
}
|
|
175
|
+
};
|
|
176
|
+
}
|
|
177
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"route.js","sourceRoot":"","sources":["../../../../../../src/api/admin/analytics/sales/route.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,204 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GET = GET;
|
|
4
|
+
const utils_1 = require("@medusajs/framework/utils");
|
|
5
|
+
async function GET(req, res) {
|
|
6
|
+
const query = req.scope.resolve(utils_1.ContainerRegistrationKeys.QUERY);
|
|
7
|
+
const productId = req.params.id;
|
|
8
|
+
// Parse query parameters
|
|
9
|
+
const startDate = req.query.start_date ? new Date(req.query.start_date) : new Date(new Date().setMonth(new Date().getMonth() - 3));
|
|
10
|
+
const endDate = req.query.end_date ? new Date(req.query.end_date) : new Date();
|
|
11
|
+
const variantId = req.query.variant_id;
|
|
12
|
+
// Get product details
|
|
13
|
+
const { data: products } = await query.graph({
|
|
14
|
+
entity: "product",
|
|
15
|
+
fields: [
|
|
16
|
+
"*",
|
|
17
|
+
"variants.*",
|
|
18
|
+
"variants.inventory_items.*",
|
|
19
|
+
"variants.calculated_price.*",
|
|
20
|
+
"collection.*",
|
|
21
|
+
"categories.*"
|
|
22
|
+
],
|
|
23
|
+
filters: {
|
|
24
|
+
id: productId
|
|
25
|
+
}
|
|
26
|
+
});
|
|
27
|
+
if (!products || products.length === 0) {
|
|
28
|
+
return res.status(404).json({ error: "Product not found" });
|
|
29
|
+
}
|
|
30
|
+
const product = products[0];
|
|
31
|
+
const variantIds = variantId
|
|
32
|
+
? [variantId]
|
|
33
|
+
: product.variants?.map((v) => v.id) || [];
|
|
34
|
+
// Get order items for this product
|
|
35
|
+
const { data: orderItems } = await query.graph({
|
|
36
|
+
entity: "order_line_item",
|
|
37
|
+
fields: [
|
|
38
|
+
"id",
|
|
39
|
+
"variant_id",
|
|
40
|
+
"quantity",
|
|
41
|
+
"unit_price",
|
|
42
|
+
"subtotal",
|
|
43
|
+
"created_at",
|
|
44
|
+
"order.*",
|
|
45
|
+
"order.customer.*"
|
|
46
|
+
],
|
|
47
|
+
filters: {
|
|
48
|
+
variant_id: variantIds
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
// Group sales by date
|
|
52
|
+
const salesByDate = {};
|
|
53
|
+
const salesByMonth = {};
|
|
54
|
+
const salesByVariant = {};
|
|
55
|
+
const customerPurchases = {};
|
|
56
|
+
orderItems.forEach(item => {
|
|
57
|
+
if (!item.order)
|
|
58
|
+
return;
|
|
59
|
+
const orderDate = new Date(item.order.created_at);
|
|
60
|
+
if (orderDate < startDate || orderDate > endDate)
|
|
61
|
+
return;
|
|
62
|
+
if (![
|
|
63
|
+
"completed",
|
|
64
|
+
"partially_returned",
|
|
65
|
+
"partially_shipped",
|
|
66
|
+
"shipped",
|
|
67
|
+
"fulfilled",
|
|
68
|
+
"partially_fulfilled"
|
|
69
|
+
].includes(item.order.status)) {
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
// Daily sales
|
|
73
|
+
const dateKey = new Date(item.order.created_at).toISOString().split("T")[0];
|
|
74
|
+
if (!salesByDate[dateKey]) {
|
|
75
|
+
salesByDate[dateKey] = {
|
|
76
|
+
date: dateKey,
|
|
77
|
+
quantity: 0,
|
|
78
|
+
revenue: 0,
|
|
79
|
+
orders: new Set()
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
salesByDate[dateKey].quantity += item.quantity || 0;
|
|
83
|
+
salesByDate[dateKey].revenue += item.subtotal || 0;
|
|
84
|
+
salesByDate[dateKey].orders.add(item.order.id);
|
|
85
|
+
// Monthly sales
|
|
86
|
+
const monthKey = new Date(item.order.created_at).toISOString().substring(0, 7);
|
|
87
|
+
if (!salesByMonth[monthKey]) {
|
|
88
|
+
salesByMonth[monthKey] = {
|
|
89
|
+
month: monthKey,
|
|
90
|
+
quantity: 0,
|
|
91
|
+
revenue: 0,
|
|
92
|
+
orders: new Set()
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
salesByMonth[monthKey].quantity += item.quantity || 0;
|
|
96
|
+
salesByMonth[monthKey].revenue += item.subtotal || 0;
|
|
97
|
+
salesByMonth[monthKey].orders.add(item.order.id);
|
|
98
|
+
// Sales by variant
|
|
99
|
+
if (!salesByVariant[item.variant_id]) {
|
|
100
|
+
const variant = product.variants?.find((v) => v.id === item.variant_id);
|
|
101
|
+
salesByVariant[item.variant_id] = {
|
|
102
|
+
variant_id: item.variant_id,
|
|
103
|
+
variant_title: variant?.title || "Unknown",
|
|
104
|
+
sku: variant?.sku,
|
|
105
|
+
quantity: 0,
|
|
106
|
+
revenue: 0,
|
|
107
|
+
orders: 0
|
|
108
|
+
};
|
|
109
|
+
}
|
|
110
|
+
salesByVariant[item.variant_id].quantity += item.quantity || 0;
|
|
111
|
+
salesByVariant[item.variant_id].revenue += item.subtotal || 0;
|
|
112
|
+
salesByVariant[item.variant_id].orders += 1;
|
|
113
|
+
// Customer purchases
|
|
114
|
+
const customerId = item.order.customer_id;
|
|
115
|
+
if (customerId) {
|
|
116
|
+
if (!customerPurchases[customerId]) {
|
|
117
|
+
customerPurchases[customerId] = {
|
|
118
|
+
customer_id: customerId,
|
|
119
|
+
customer_email: item.order.customer?.email,
|
|
120
|
+
customer_name: `${item.order.customer?.first_name || ''} ${item.order.customer?.last_name || ''}`.trim(),
|
|
121
|
+
total_purchases: 0,
|
|
122
|
+
total_spent: 0,
|
|
123
|
+
first_purchase: item.order.created_at,
|
|
124
|
+
last_purchase: item.order.created_at
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
customerPurchases[customerId].total_purchases += item.quantity || 0;
|
|
128
|
+
customerPurchases[customerId].total_spent += item.subtotal || 0;
|
|
129
|
+
if (new Date(item.order.created_at) < new Date(customerPurchases[customerId].first_purchase)) {
|
|
130
|
+
customerPurchases[customerId].first_purchase = item.order.created_at;
|
|
131
|
+
}
|
|
132
|
+
if (new Date(item.order.created_at) > new Date(customerPurchases[customerId].last_purchase)) {
|
|
133
|
+
customerPurchases[customerId].last_purchase = item.order.created_at;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
});
|
|
137
|
+
// Convert to arrays and sort
|
|
138
|
+
const dailySales = Object.keys(salesByDate)
|
|
139
|
+
.sort()
|
|
140
|
+
.map(key => ({
|
|
141
|
+
...salesByDate[key],
|
|
142
|
+
order_count: salesByDate[key].orders.size
|
|
143
|
+
}));
|
|
144
|
+
const monthlySales = Object.keys(salesByMonth)
|
|
145
|
+
.sort()
|
|
146
|
+
.map(key => ({
|
|
147
|
+
...salesByMonth[key],
|
|
148
|
+
order_count: salesByMonth[key].orders.size
|
|
149
|
+
}));
|
|
150
|
+
const variantSales = Object.values(salesByVariant)
|
|
151
|
+
.sort((a, b) => b.revenue - a.revenue);
|
|
152
|
+
const topCustomers = Object.values(customerPurchases)
|
|
153
|
+
.sort((a, b) => b.total_spent - a.total_spent)
|
|
154
|
+
.slice(0, 10);
|
|
155
|
+
// Calculate summary
|
|
156
|
+
const totalQuantity = orderItems.reduce((sum, item) => sum + (item.quantity || 0), 0);
|
|
157
|
+
const totalRevenue = orderItems.reduce((sum, item) => sum + (item.subtotal || 0), 0);
|
|
158
|
+
// Get current inventory
|
|
159
|
+
const currentInventory = product.variants?.reduce((total, variant) => {
|
|
160
|
+
const inventory = variant.inventory_items?.[0]?.inventory ||
|
|
161
|
+
variant.inventory_items?.[0];
|
|
162
|
+
return total + (inventory?.available_quantity || 0);
|
|
163
|
+
}, 0) || 0;
|
|
164
|
+
res.json({
|
|
165
|
+
product: {
|
|
166
|
+
id: product.id,
|
|
167
|
+
title: product.title,
|
|
168
|
+
handle: product.handle,
|
|
169
|
+
collection: product.collection,
|
|
170
|
+
categories: product.categories
|
|
171
|
+
},
|
|
172
|
+
sales_history: {
|
|
173
|
+
period: {
|
|
174
|
+
start_date: startDate,
|
|
175
|
+
end_date: endDate
|
|
176
|
+
},
|
|
177
|
+
summary: {
|
|
178
|
+
total_quantity_sold: totalQuantity,
|
|
179
|
+
total_revenue: totalRevenue,
|
|
180
|
+
unique_customers: Object.keys(customerPurchases).length,
|
|
181
|
+
current_inventory: currentInventory,
|
|
182
|
+
average_sale_price: totalQuantity > 0 ? totalRevenue / totalQuantity : 0
|
|
183
|
+
},
|
|
184
|
+
daily_sales: dailySales,
|
|
185
|
+
monthly_sales: monthlySales,
|
|
186
|
+
variant_breakdown: variantSales,
|
|
187
|
+
top_customers: topCustomers,
|
|
188
|
+
sales_velocity: calculateVelocity(dailySales),
|
|
189
|
+
inventory_turnover: currentInventory > 0 ? totalQuantity / currentInventory : 0
|
|
190
|
+
}
|
|
191
|
+
});
|
|
192
|
+
}
|
|
193
|
+
function calculateVelocity(dailySales) {
|
|
194
|
+
if (dailySales.length === 0)
|
|
195
|
+
return 0;
|
|
196
|
+
const totalDays = dailySales.length;
|
|
197
|
+
const totalQuantity = dailySales.reduce((sum, day) => sum + day.quantity, 0);
|
|
198
|
+
return {
|
|
199
|
+
daily_average: totalQuantity / totalDays,
|
|
200
|
+
weekly_average: (totalQuantity / totalDays) * 7,
|
|
201
|
+
monthly_average: (totalQuantity / totalDays) * 30
|
|
202
|
+
};
|
|
203
|
+
}
|
|
204
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"route.js","sourceRoot":"","sources":["../../../../../../../src/api/admin/products/[id]/sales-history/route.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GET = GET;
|
|
4
|
+
const utils_1 = require("@medusajs/framework/utils");
|
|
5
|
+
async function GET(req, res) {
|
|
6
|
+
const query = req.scope.resolve(utils_1.ContainerRegistrationKeys.QUERY);
|
|
7
|
+
const { data: products } = await query.graph({
|
|
8
|
+
entity: "product",
|
|
9
|
+
fields: [
|
|
10
|
+
"*",
|
|
11
|
+
"categories.*",
|
|
12
|
+
"collection.*",
|
|
13
|
+
"tags.*",
|
|
14
|
+
"type.*",
|
|
15
|
+
"images.*",
|
|
16
|
+
"variants.*",
|
|
17
|
+
"variants.inventory_items.*",
|
|
18
|
+
"variants.calculated_price.*",
|
|
19
|
+
"variants.prices.*",
|
|
20
|
+
"options.*",
|
|
21
|
+
"options.values.*"
|
|
22
|
+
],
|
|
23
|
+
filters: req.query.filters || {}
|
|
24
|
+
});
|
|
25
|
+
// First get all order items with order details
|
|
26
|
+
const { data: orderItems } = await query.graph({
|
|
27
|
+
entity: "order_line_item",
|
|
28
|
+
fields: [
|
|
29
|
+
"id",
|
|
30
|
+
"variant_id",
|
|
31
|
+
"quantity",
|
|
32
|
+
"unit_price",
|
|
33
|
+
"subtotal",
|
|
34
|
+
"order.*"
|
|
35
|
+
]
|
|
36
|
+
});
|
|
37
|
+
// Filter for completed orders
|
|
38
|
+
const filteredOrderItems = orderItems.filter(item => item.order && [
|
|
39
|
+
"completed",
|
|
40
|
+
"partially_returned",
|
|
41
|
+
"partially_shipped",
|
|
42
|
+
"shipped"
|
|
43
|
+
].includes(item.order.status));
|
|
44
|
+
const salesByProduct = products.map(product => {
|
|
45
|
+
const variantIds = product.variants?.map(v => v.id) || [];
|
|
46
|
+
const productSales = filteredOrderItems.filter(item => item.variant_id && variantIds.includes(item.variant_id));
|
|
47
|
+
const totalQuantitySold = productSales.reduce((sum, item) => sum + (item.quantity || 0), 0);
|
|
48
|
+
const totalRevenue = productSales.reduce((sum, item) => sum + (item.subtotal || 0), 0);
|
|
49
|
+
const salesByVariant = product.variants?.map(variant => {
|
|
50
|
+
const variantSales = productSales.filter(item => item.variant_id === variant.id);
|
|
51
|
+
const variantQuantitySold = variantSales.reduce((sum, item) => sum + (item.quantity || 0), 0);
|
|
52
|
+
const variantRevenue = variantSales.reduce((sum, item) => sum + (item.subtotal || 0), 0);
|
|
53
|
+
return {
|
|
54
|
+
variant_id: variant.id,
|
|
55
|
+
variant_title: variant.title,
|
|
56
|
+
sku: variant.sku,
|
|
57
|
+
barcode: variant.barcode,
|
|
58
|
+
quantity_sold: variantQuantitySold,
|
|
59
|
+
revenue: variantRevenue,
|
|
60
|
+
stock_quantity: variant.inventory_items?.[0]?.inventory?.stocked_quantity || 0,
|
|
61
|
+
available_quantity: variant.inventory_items?.[0]?.inventory?.available_quantity || 0,
|
|
62
|
+
reserved_quantity: variant.inventory_items?.[0]?.inventory?.reserved_quantity || 0,
|
|
63
|
+
calculated_price: variant.calculated_price,
|
|
64
|
+
prices: variant.prices,
|
|
65
|
+
weight: variant.weight,
|
|
66
|
+
length: variant.length,
|
|
67
|
+
height: variant.height,
|
|
68
|
+
width: variant.width,
|
|
69
|
+
material: variant.material,
|
|
70
|
+
metadata: variant.metadata,
|
|
71
|
+
manage_inventory: variant.manage_inventory,
|
|
72
|
+
allow_backorder: variant.allow_backorder
|
|
73
|
+
};
|
|
74
|
+
}) || [];
|
|
75
|
+
return {
|
|
76
|
+
...product,
|
|
77
|
+
total_quantity_sold: totalQuantitySold,
|
|
78
|
+
total_revenue: totalRevenue,
|
|
79
|
+
number_of_orders: new Set(productSales.map((item) => item.order?.id).filter(Boolean)).size,
|
|
80
|
+
variants_with_sales: salesByVariant,
|
|
81
|
+
average_order_value: totalQuantitySold > 0 ? totalRevenue / totalQuantitySold : 0,
|
|
82
|
+
is_bestseller: totalQuantitySold > 0,
|
|
83
|
+
stock_status: product.variants?.some((v) => v.inventory_items?.[0]?.inventory?.available_quantity > 0) ? 'in_stock' : 'out_of_stock'
|
|
84
|
+
};
|
|
85
|
+
});
|
|
86
|
+
salesByProduct.sort((a, b) => b.total_quantity_sold - a.total_quantity_sold);
|
|
87
|
+
res.json({
|
|
88
|
+
statistics: salesByProduct,
|
|
89
|
+
summary: {
|
|
90
|
+
total_products: products.length,
|
|
91
|
+
products_with_sales: salesByProduct.filter(p => p.total_quantity_sold > 0).length,
|
|
92
|
+
grand_total_revenue: salesByProduct.reduce((sum, p) => sum + p.total_revenue, 0),
|
|
93
|
+
grand_total_quantity: salesByProduct.reduce((sum, p) => sum + p.total_quantity_sold, 0)
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm91dGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvYXBpL2FkbWluL3Byb2R1Y3RzL3NhbGVzLXN0YXRpc3RpY3Mvcm91dGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFHQSxrQkE0SEM7QUE5SEQscURBQXNFO0FBRS9ELEtBQUssVUFBVSxHQUFHLENBQ3ZCLEdBQWtCLEVBQ2xCLEdBQW1CO0lBRW5CLE1BQU0sS0FBSyxHQUFHLEdBQUcsQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLGlDQUF5QixDQUFDLEtBQUssQ0FBQyxDQUFDO0lBRWpFLE1BQU0sRUFBRSxJQUFJLEVBQUUsUUFBUSxFQUFFLEdBQUcsTUFBTSxLQUFLLENBQUMsS0FBSyxDQUFDO1FBQzNDLE1BQU0sRUFBRSxTQUFTO1FBQ2pCLE1BQU0sRUFBRTtZQUNOLEdBQUc7WUFDSCxjQUFjO1lBQ2QsY0FBYztZQUNkLFFBQVE7WUFDUixRQUFRO1lBQ1IsVUFBVTtZQUNWLFlBQVk7WUFDWiw0QkFBNEI7WUFDNUIsNkJBQTZCO1lBQzdCLG1CQUFtQjtZQUNuQixXQUFXO1lBQ1gsa0JBQWtCO1NBQ25CO1FBQ0QsT0FBTyxFQUFHLEdBQUcsQ0FBQyxLQUFLLENBQUMsT0FBZSxJQUFJLEVBQUU7S0FDMUMsQ0FBQyxDQUFDO0lBRUgsK0NBQStDO0lBQy9DLE1BQU0sRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFLEdBQUcsTUFBTSxLQUFLLENBQUMsS0FBSyxDQUFDO1FBQzdDLE1BQU0sRUFBRSxpQkFBaUI7UUFDekIsTUFBTSxFQUFFO1lBQ04sSUFBSTtZQUNKLFlBQVk7WUFDWixVQUFVO1lBQ1YsWUFBWTtZQUNaLFVBQVU7WUFDVixTQUFTO1NBQ1Y7S0FDRixDQUFvQixDQUFDO0lBRXRCLDhCQUE4QjtJQUM5QixNQUFNLGtCQUFrQixHQUFHLFVBQVUsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FDbEQsSUFBSSxDQUFDLEtBQUssSUFB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
|