@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.
@@ -0,0 +1,2 @@
1
+ import { MedusaRequest, MedusaResponse } from "@medusajs/framework/http";
2
+ export declare function GET(req: MedusaRequest, res: MedusaResponse): Promise<void>;
@@ -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":";;AAGA,kBAmKC;AArKD,qDAAsE;AAE/D,KAAK,UAAU,GAAG,CACvB,GAAkB,EAClB,GAAmB;IAEnB,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,iCAAyB,CAAC,KAAK,CAAC,CAAC;IAEjE,yBAAyB;IACzB,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,UAAoB,CAAC,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC,QAAQ,CAAC,IAAI,IAAI,EAAE,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC;IAC7I,MAAM,OAAO,GAAG,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,QAAkB,CAAC,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CAAC;IACzF,MAAM,OAAO,GAAG,GAAG,CAAC,KAAK,CAAC,QAAkB,IAAI,KAAK,CAAC,CAAC,yBAAyB;IAChF,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,UAAoB,CAAC;IACjD,MAAM,YAAY,GAAG,GAAG,CAAC,KAAK,CAAC,aAAuB,CAAC;IACvD,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,WAAqB,CAAC;IAEnD,gBAAgB;IAChB,IAAI,OAAO,GAAQ,EAAE,CAAC;IACtB,IAAI,SAAS;QAAE,OAAO,CAAC,UAAU,GAAG,SAAS,CAAC;IAC9C,IAAI,YAAY;QAAE,OAAO,CAAC,aAAa,GAAG,YAAY,CAAC;IACvD,IAAI,UAAU;QAAE,OAAO,CAAC,WAAW,GAAG,UAAU,CAAC;IAEjD,gCAAgC;IAChC,MAAM,EAAE,IAAI,EAAE,QAAQ,EAAE,GAAG,MAAM,KAAK,CAAC,KAAK,CAAC;QAC3C,MAAM,EAAE,SAAS;QACjB,MAAM,EAAE,CAAC,IAAI,EAAE,OAAO,EAAE,YAAY,EAAE,cAAc,EAAE,cAAc,CAAC;QACrE,OAAO,EAAE,OAAO;KACjB,CAAC,CAAC;IAEH,oCAAoC;IACpC,MAAM,EAAE,IAAI,EAAE,UAAU,EAAE,GAAG,MAAM,KAAK,CAAC,KAAK,CAAC;QAC7C,MAAM,EAAE,iBAAiB;QACzB,MAAM,EAAE;YACN,IAAI;YACJ,YAAY;YACZ,YAAY;YACZ,UAAU;YACV,YAAY;YACZ,UAAU;YACV,SAAS;SACV;KACF,CAAoB,CAAC;IAEtB,6CAA6C;IAC7C,MAAM,mBAAmB,GAAG,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE;QACnD,IAAI,CAAC,IAAI,CAAC,KAAK;YAAE,OAAO,KAAK,CAAC;QAC9B,MAAM,SAAS,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAClD,OAAO,SAAS,IAAI,SAAS,IAAI,SAAS,IAAI,OAAO,IAAI;YACvD,WAAW;YACX,oBAAoB;YACpB,mBAAmB;YACnB,SAAS;YACT,WAAW;YACX,qBAAqB;SACtB,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IAChC,CAAC,CAAC,CAAC;IAEH,wBAAwB;IACxB,MAAM,aAAa,GAA2B,EAAE,CAAC;IAEjD,mBAAmB,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;QACjC,MAAM,SAAS,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAClD,IAAI,SAAiB,CAAC;QAEtB,QAAO,OAAO,EAAE,CAAC;YACf,KAAK,KAAK;gBACR,SAAS,GAAG,SAAS,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;gBAClD,MAAM;YACR,KAAK,MAAM;gBACT,MAAM,SAAS,GAAG,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC;gBACtC,SAAS,CAAC,OAAO,CAAC,SAAS,CAAC,OAAO,EAAE,GAAG,SAAS,CAAC,MAAM,EAAE,CAAC,CAAC;gBAC5D,SAAS,GAAG,SAAS,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;gBAClD,MAAM;YACR,KAAK,OAAO;gBACV,SAAS,GAAG,GAAG,SAAS,CAAC,WAAW,EAAE,IAAI,MAAM,CAAC,SAAS,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;gBAC9F,MAAM;YACR,KAAK,MAAM;gBACT,SAAS,GAAG,SAAS,CAAC,WAAW,EAAE,CAAC,QAAQ,EAAE,CAAC;gBAC/C,MAAM;YACR;gBACE,SAAS,GAAG,SAAS,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;QACtD,CAAC;QAED,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,EAAE,CAAC;YAC9B,aAAa,CAAC,SAAS,CAAC,GAAG;gBACzB,IAAI,EAAE,SAAS;gBACf,OAAO,EAAE,CAAC;gBACV,QAAQ,EAAE,CAAC;gBACX,MAAM,EAAE,IAAI,GAAG,EAAE;gBACjB,aAAa,EAAE,IAAI,GAAG,EAAE;aACzB,CAAC;QACJ,CAAC;QAED,aAAa,CAAC,SAAS,CAAC,CAAC,OAAO,IAAI,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC;QACvD,aAAa,CAAC,SAAS,CAAC,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC;QACxD,aAAa,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACnD,aAAa,CAAC,SAAS,CAAC,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC9D,CAAC,CAAC,CAAC;IAEH,+CAA+C;IAC/C,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC;SACzC,IAAI,EAAE;SACN,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACX,IAAI,EAAE,GAAG;QACT,OAAO,EAAE,aAAa,CAAC,GAAG,CAAC,CAAC,OAAO;QACnC,QAAQ,EAAE,aAAa,CAAC,GAAG,CAAC,CAAC,QAAQ;QACrC,WAAW,EAAE,aAAa,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,IAAI;QAC3C,eAAe,EAAE,aAAa,CAAC,GAAG,CAAC,CAAC,aAAa,CAAC,IAAI;QACtD,mBAAmB,EAAE,aAAa,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC;YACrD,CAAC,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,OAAO,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,IAAI;YAC7D,CAAC,CAAC,CAAC;KACN,CAAC,CAAC,CAAC;IAEN,+BAA+B;IAC/B,MAAM,YAAY,GAAG,mBAAmB,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9F,MAAM,aAAa,GAAG,mBAAmB,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAC/F,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,mBAAmB,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9F,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC,mBAAmB,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC;IAE1G,eAAe;IACf,MAAM,YAAY,GAA2B,EAAE,CAAC;IAEhD,mBAAmB,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;QACjC,MAAM,OAAO,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAChC,CAAC,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,IAAI,CAAC,UAAU,CAAC,CACvD,CAAC;QAEF,IAAI,OAAO,EAAE,CAAC;YACZ,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC;gBAC9B,YAAY,CAAC,OAAO,CAAC,EAAE,CAAC,GAAG;oBACzB,UAAU,EAAE,OAAO,CAAC,EAAE;oBACtB,aAAa,EAAE,OAAO,CAAC,KAAK;oBAC5B,aAAa,EAAE,CAAC;oBAChB,OAAO,EAAE,CAAC;iBACX,CAAC;YACJ,CAAC;YACD,YAAY,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,aAAa,IAAI,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC;YAC7D,YAAY,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,OAAO,IAAI,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC;QACzD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,MAAM,CAAC,MAAM,CAAC,YAAY,CAAC;SAC5C,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC,OAAO,CAAC;SACrC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IAEhB,GAAG,CAAC,IAAI,CAAC;QACP,SAAS,EAAE;YACT,MAAM,EAAE;gBACN,UAAU,EAAE,SAAS;gBACrB,QAAQ,EAAE,OAAO;gBACjB,QAAQ,EAAE,OAAO;aAClB;YACD,OAAO,EAAE;gBACP,aAAa,EAAE,YAAY;gBAC3B,mBAAmB,EAAE,aAAa;gBAClC,YAAY,EAAE,YAAY,CAAC,IAAI;gBAC/B,gBAAgB,EAAE,eAAe,CAAC,IAAI;gBACtC,mBAAmB,EAAE,YAAY,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC,YAAY,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;gBACjF,uBAAuB,EAAE,YAAY,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC,aAAa,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;aACvF;YACD,UAAU,EAAE,SAAS;YACrB,YAAY,EAAE,WAAW;YACzB,MAAM,EAAE,eAAe,CAAC,SAAS,CAAC;SACnC;KACF,CAAC,CAAC;AACL,CAAC;AAED,SAAS,eAAe,CAAC,IAAW;IAClC,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC;QAAE,OAAO,IAAI,CAAC;IAEjC,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IACtC,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IAEvC,OAAO;QACL,OAAO,EAAE;YACP,KAAK,EAAE,OAAO,CAAC,OAAO,GAAG,QAAQ,CAAC,OAAO;YACzC,UAAU,EAAE,QAAQ,CAAC,OAAO,GAAG,CAAC;gBAC9B,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,QAAQ,CAAC,OAAO,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;gBAC5E,CAAC,CAAC,CAAC;SACN;QACD,QAAQ,EAAE;YACR,KAAK,EAAE,OAAO,CAAC,QAAQ,GAAG,QAAQ,CAAC,QAAQ;YAC3C,UAAU,EAAE,QAAQ,CAAC,QAAQ,GAAG,CAAC;gBAC/B,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,GAAG,QAAQ,CAAC,QAAQ,CAAC,GAAG,QAAQ,CAAC,QAAQ,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;gBAC/E,CAAC,CAAC,CAAC;SACN;QACD,MAAM,EAAE;YACN,KAAK,EAAE,OAAO,CAAC,WAAW,GAAG,QAAQ,CAAC,WAAW;YACjD,UAAU,EAAE,QAAQ,CAAC,WAAW,GAAG,CAAC;gBAClC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,WAAW,GAAG,QAAQ,CAAC,WAAW,CAAC,GAAG,QAAQ,CAAC,WAAW,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;gBACxF,CAAC,CAAC,CAAC;SACN;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ import { MedusaRequest, MedusaResponse } from "@medusajs/framework/http";
2
+ export declare function GET(req: MedusaRequest, res: MedusaResponse): Promise<MedusaResponse>;
@@ -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,2 @@
1
+ import { MedusaRequest, MedusaResponse } from "@medusajs/framework/http";
2
+ export declare function GET(req: MedusaRequest, res: MedusaResponse): Promise<void>;
@@ -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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
@@ -0,0 +1,2 @@
1
+ import { MedusaRequest, MedusaResponse } from "@medusajs/framework/http";
2
+ export declare function GET(req: MedusaRequest, res: MedusaResponse): Promise<void>;