From 2601a0421125357f7c3b1dfe98fb98717451d70d Mon Sep 17 00:00:00 2001
From: Matt
Date: Wed, 2 Apr 2025 15:42:20 -0400
Subject: [PATCH] Category calculation fixes
---
inventory-server/db/metrics-schema-new.sql | 20 +-
.../scripts/calculate-metrics-new.js | 4 +-
.../calculate_category_metrics.sql | 219 +++++++++++++-----
inventory/src/pages/Categories.tsx | 183 ++++++++++-----
4 files changed, 304 insertions(+), 122 deletions(-)
diff --git a/inventory-server/db/metrics-schema-new.sql b/inventory-server/db/metrics-schema-new.sql
index 27a7df8..3c7516a 100644
--- a/inventory-server/db/metrics-schema-new.sql
+++ b/inventory-server/db/metrics-schema-new.sql
@@ -178,6 +178,7 @@ CREATE TABLE public.category_metrics (
parent_id INT8, -- Denormalized for convenience
last_calculated TIMESTAMPTZ NOT NULL DEFAULT NOW(),
+ -- ROLLED-UP METRICS (includes this category + all descendants)
-- Counts & Basic Info
product_count INT NOT NULL DEFAULT 0, -- Total products linked
active_product_count INT NOT NULL DEFAULT 0, -- Visible products linked
@@ -195,7 +196,24 @@ CREATE TABLE public.category_metrics (
sales_365d INT NOT NULL DEFAULT 0, revenue_365d NUMERIC(16, 4) NOT NULL DEFAULT 0.00,
lifetime_sales INT NOT NULL DEFAULT 0, lifetime_revenue NUMERIC(18, 4) NOT NULL DEFAULT 0.00,
- -- Calculated KPIs (Based on 30d aggregates)
+ -- DIRECT METRICS (only products directly in this category)
+ direct_product_count INT NOT NULL DEFAULT 0, -- Products directly in this category
+ direct_active_product_count INT NOT NULL DEFAULT 0, -- Visible products directly in this category
+ direct_replenishable_product_count INT NOT NULL DEFAULT 0,-- Replenishable products directly in this category
+
+ -- Direct Current Stock Value
+ direct_current_stock_units INT NOT NULL DEFAULT 0,
+ direct_stock_cost NUMERIC(16, 4) NOT NULL DEFAULT 0.00,
+ direct_stock_retail NUMERIC(16, 4) NOT NULL DEFAULT 0.00,
+
+ -- Direct Rolling Period Aggregates
+ direct_sales_7d INT NOT NULL DEFAULT 0, direct_revenue_7d NUMERIC(16, 4) NOT NULL DEFAULT 0.00,
+ direct_sales_30d INT NOT NULL DEFAULT 0, direct_revenue_30d NUMERIC(16, 4) NOT NULL DEFAULT 0.00,
+ direct_profit_30d NUMERIC(16, 4) NOT NULL DEFAULT 0.00, direct_cogs_30d NUMERIC(16, 4) NOT NULL DEFAULT 0.00,
+ direct_sales_365d INT NOT NULL DEFAULT 0, direct_revenue_365d NUMERIC(16, 4) NOT NULL DEFAULT 0.00,
+ direct_lifetime_sales INT NOT NULL DEFAULT 0, direct_lifetime_revenue NUMERIC(18, 4) NOT NULL DEFAULT 0.00,
+
+ -- Calculated KPIs (Based on 30d aggregates) - Apply to rolled-up metrics
avg_margin_30d NUMERIC(7, 3), -- (profit / revenue) * 100
stock_turn_30d NUMERIC(10, 3), -- sales_units / avg_stock_units (Needs avg stock calc)
-- growth_rate_30d NUMERIC(7, 3), -- (current 30d rev - prev 30d rev) / prev 30d rev
diff --git a/inventory-server/scripts/calculate-metrics-new.js b/inventory-server/scripts/calculate-metrics-new.js
index 2dcc9b1..b53ae99 100644
--- a/inventory-server/scripts/calculate-metrics-new.js
+++ b/inventory-server/scripts/calculate-metrics-new.js
@@ -8,8 +8,8 @@ const { Pool } = require('pg'); // Assuming you use 'pg'
const RUN_DAILY_SNAPSHOTS = false;
const RUN_PRODUCT_METRICS = false;
const RUN_PERIODIC_METRICS = false;
-const RUN_BRAND_METRICS = true;
-const RUN_VENDOR_METRICS = true;
+const RUN_BRAND_METRICS = false;
+const RUN_VENDOR_METRICS = false;
const RUN_CATEGORY_METRICS = true;
// Maximum execution time for the entire sequence (e.g., 90 minutes)
diff --git a/inventory-server/scripts/metrics-new/calculate_category_metrics.sql b/inventory-server/scripts/metrics-new/calculate_category_metrics.sql
index 6427bcf..995278d 100644
--- a/inventory-server/scripts/metrics-new/calculate_category_metrics.sql
+++ b/inventory-server/scripts/metrics-new/calculate_category_metrics.sql
@@ -6,6 +6,7 @@ DO $$
DECLARE
_module_name VARCHAR := 'category_metrics';
_start_time TIMESTAMPTZ := clock_timestamp();
+ _min_revenue NUMERIC := 50.00; -- Minimum revenue threshold for margin calculation
BEGIN
RAISE NOTICE 'Running % calculation...', _module_name;
@@ -44,7 +45,8 @@ BEGIN
)
),
-- Calculate metrics only at the most specific category level for each product
- CategoryAggregates AS (
+ -- These are the direct metrics (only products directly in this category)
+ DirectCategoryMetrics AS (
SELECT
pdc.cat_id,
-- Counts
@@ -72,95 +74,171 @@ BEGIN
JOIN ProductDeepestCategory pdc ON pm.pid = pdc.pid
GROUP BY pdc.cat_id
),
- -- Use a flat approach to build the complete category tree with aggregate values
- CategoryTree AS (
- WITH RECURSIVE CategoryHierarchy AS (
+ -- Build a category lookup table for parent relationships
+ CategoryHierarchyPaths AS (
+ WITH RECURSIVE ParentPaths AS (
+ -- Base case: All categories with their immediate parents
SELECT
- c.cat_id,
- c.name,
- c.parent_id,
- c.cat_id as leaf_id, -- Track original leaf category
- ARRAY[c.cat_id] as path
- FROM public.categories c
+ cat_id,
+ cat_id as leaf_id, -- Every category is its own leaf initially
+ ARRAY[cat_id] as path
+ FROM public.categories
UNION ALL
+ -- Recursive step: Walk up the parent chain
SELECT
- p.cat_id,
- p.name,
- p.parent_id,
- ch.leaf_id, -- Keep track of the original leaf
- p.cat_id || ch.path
- FROM public.categories p
- JOIN CategoryHierarchy ch ON p.cat_id = ch.parent_id
+ c.parent_id as cat_id,
+ pp.leaf_id, -- Keep the original leaf_id
+ c.parent_id || pp.path as path
+ FROM ParentPaths pp
+ JOIN public.categories c ON pp.cat_id = c.cat_id
+ WHERE c.parent_id IS NOT NULL -- Stop at root categories
)
- SELECT
- ch.cat_id,
- ch.leaf_id
- FROM CategoryHierarchy ch
+ -- Select distinct paths to avoid duplication
+ SELECT DISTINCT cat_id, leaf_id
+ FROM ParentPaths
),
- -- Now aggregate by maintaining the link between leaf categories and ancestors
+ -- Aggregate metrics from leaf categories to their ancestors without duplication
+ -- These are the rolled-up metrics (including all child categories)
RollupMetrics AS (
SELECT
- ct.cat_id,
- SUM(ca.product_count) AS product_count,
- SUM(ca.active_product_count) AS active_product_count,
- SUM(ca.replenishable_product_count) AS replenishable_product_count,
- SUM(ca.current_stock_units) AS current_stock_units,
- SUM(ca.current_stock_cost) AS current_stock_cost,
- SUM(ca.current_stock_retail) AS current_stock_retail,
- SUM(ca.sales_7d) AS sales_7d,
- SUM(ca.revenue_7d) AS revenue_7d,
- SUM(ca.sales_30d) AS sales_30d,
- SUM(ca.revenue_30d) AS revenue_30d,
- SUM(ca.cogs_30d) AS cogs_30d,
- SUM(ca.profit_30d) AS profit_30d,
- SUM(ca.sales_365d) AS sales_365d,
- SUM(ca.revenue_365d) AS revenue_365d,
- SUM(ca.lifetime_sales) AS lifetime_sales,
- SUM(ca.lifetime_revenue) AS lifetime_revenue,
- SUM(ca.total_avg_stock_units_30d) AS total_avg_stock_units_30d
- FROM CategoryTree ct
- JOIN CategoryAggregates ca ON ct.leaf_id = ca.cat_id
- GROUP BY ct.cat_id
+ chp.cat_id,
+ -- For each parent category, count distinct products to avoid duplication
+ COUNT(DISTINCT dcm.cat_id) AS child_categories_count,
+ SUM(dcm.product_count) AS rollup_product_count,
+ SUM(dcm.active_product_count) AS rollup_active_product_count,
+ SUM(dcm.replenishable_product_count) AS rollup_replenishable_product_count,
+ SUM(dcm.current_stock_units) AS rollup_current_stock_units,
+ SUM(dcm.current_stock_cost) AS rollup_current_stock_cost,
+ SUM(dcm.current_stock_retail) AS rollup_current_stock_retail,
+ SUM(dcm.sales_7d) AS rollup_sales_7d,
+ SUM(dcm.revenue_7d) AS rollup_revenue_7d,
+ SUM(dcm.sales_30d) AS rollup_sales_30d,
+ SUM(dcm.revenue_30d) AS rollup_revenue_30d,
+ SUM(dcm.cogs_30d) AS rollup_cogs_30d,
+ SUM(dcm.profit_30d) AS rollup_profit_30d,
+ SUM(dcm.sales_365d) AS rollup_sales_365d,
+ SUM(dcm.revenue_365d) AS rollup_revenue_365d,
+ SUM(dcm.lifetime_sales) AS rollup_lifetime_sales,
+ SUM(dcm.lifetime_revenue) AS rollup_lifetime_revenue,
+ SUM(dcm.total_avg_stock_units_30d) AS rollup_total_avg_stock_units_30d
+ FROM CategoryHierarchyPaths chp
+ JOIN DirectCategoryMetrics dcm ON chp.leaf_id = dcm.cat_id
+ GROUP BY chp.cat_id
+ ),
+ -- Combine direct and rollup metrics
+ CombinedMetrics AS (
+ SELECT
+ c.cat_id,
+ c.name,
+ c.type,
+ c.parent_id,
+ -- Direct metrics (just this category)
+ COALESCE(dcm.product_count, 0) AS direct_product_count,
+ COALESCE(dcm.active_product_count, 0) AS direct_active_product_count,
+ COALESCE(dcm.replenishable_product_count, 0) AS direct_replenishable_product_count,
+ COALESCE(dcm.current_stock_units, 0) AS direct_current_stock_units,
+ COALESCE(dcm.current_stock_cost, 0) AS direct_current_stock_cost,
+ COALESCE(dcm.current_stock_retail, 0) AS direct_current_stock_retail,
+ COALESCE(dcm.sales_7d, 0) AS direct_sales_7d,
+ COALESCE(dcm.revenue_7d, 0) AS direct_revenue_7d,
+ COALESCE(dcm.sales_30d, 0) AS direct_sales_30d,
+ COALESCE(dcm.revenue_30d, 0) AS direct_revenue_30d,
+ COALESCE(dcm.cogs_30d, 0) AS direct_cogs_30d,
+ COALESCE(dcm.profit_30d, 0) AS direct_profit_30d,
+ COALESCE(dcm.sales_365d, 0) AS direct_sales_365d,
+ COALESCE(dcm.revenue_365d, 0) AS direct_revenue_365d,
+ COALESCE(dcm.lifetime_sales, 0) AS direct_lifetime_sales,
+ COALESCE(dcm.lifetime_revenue, 0) AS direct_lifetime_revenue,
+ COALESCE(dcm.total_avg_stock_units_30d, 0) AS direct_avg_stock_units_30d,
+
+ -- Rolled up metrics (this category + all children)
+ COALESCE(rm.rollup_product_count, 0) AS product_count,
+ COALESCE(rm.rollup_active_product_count, 0) AS active_product_count,
+ COALESCE(rm.rollup_replenishable_product_count, 0) AS replenishable_product_count,
+ COALESCE(rm.rollup_current_stock_units, 0) AS current_stock_units,
+ COALESCE(rm.rollup_current_stock_cost, 0) AS current_stock_cost,
+ COALESCE(rm.rollup_current_stock_retail, 0) AS current_stock_retail,
+ COALESCE(rm.rollup_sales_7d, 0) AS sales_7d,
+ COALESCE(rm.rollup_revenue_7d, 0) AS revenue_7d,
+ COALESCE(rm.rollup_sales_30d, 0) AS sales_30d,
+ COALESCE(rm.rollup_revenue_30d, 0) AS revenue_30d,
+ COALESCE(rm.rollup_cogs_30d, 0) AS cogs_30d,
+ COALESCE(rm.rollup_profit_30d, 0) AS profit_30d,
+ COALESCE(rm.rollup_sales_365d, 0) AS sales_365d,
+ COALESCE(rm.rollup_revenue_365d, 0) AS revenue_365d,
+ COALESCE(rm.rollup_lifetime_sales, 0) AS lifetime_sales,
+ COALESCE(rm.rollup_lifetime_revenue, 0) AS lifetime_revenue,
+ COALESCE(rm.rollup_total_avg_stock_units_30d, 0) AS total_avg_stock_units_30d
+ FROM public.categories c
+ LEFT JOIN DirectCategoryMetrics dcm ON c.cat_id = dcm.cat_id
+ LEFT JOIN RollupMetrics rm ON c.cat_id = rm.cat_id
)
INSERT INTO public.category_metrics (
category_id, category_name, category_type, parent_id, last_calculated,
+ -- Store all direct and rolled up metrics
product_count, active_product_count, replenishable_product_count,
current_stock_units, current_stock_cost, current_stock_retail,
sales_7d, revenue_7d, sales_30d, revenue_30d, profit_30d, cogs_30d,
sales_365d, revenue_365d, lifetime_sales, lifetime_revenue,
+ -- Also store direct metrics with direct_ prefix
+ direct_product_count, direct_active_product_count, direct_replenishable_product_count,
+ direct_current_stock_units, direct_stock_cost, direct_stock_retail,
+ direct_sales_7d, direct_revenue_7d, direct_sales_30d, direct_revenue_30d,
+ direct_profit_30d, direct_cogs_30d, direct_sales_365d, direct_revenue_365d,
+ direct_lifetime_sales, direct_lifetime_revenue,
+ -- KPIs
avg_margin_30d, stock_turn_30d
)
SELECT
- c.cat_id,
- c.name,
- c.type,
- c.parent_id,
+ cm.cat_id,
+ cm.name,
+ cm.type,
+ cm.parent_id,
_start_time,
- -- Base Aggregates
- COALESCE(rm.product_count, 0),
- COALESCE(rm.active_product_count, 0),
- COALESCE(rm.replenishable_product_count, 0),
- COALESCE(rm.current_stock_units, 0),
- COALESCE(rm.current_stock_cost, 0.00),
- COALESCE(rm.current_stock_retail, 0.00),
- COALESCE(rm.sales_7d, 0), COALESCE(rm.revenue_7d, 0.00),
- COALESCE(rm.sales_30d, 0), COALESCE(rm.revenue_30d, 0.00),
- COALESCE(rm.profit_30d, 0.00), COALESCE(rm.cogs_30d, 0.00),
- COALESCE(rm.sales_365d, 0), COALESCE(rm.revenue_365d, 0.00),
- COALESCE(rm.lifetime_sales, 0), COALESCE(rm.lifetime_revenue, 0.00),
- -- KPIs
- (rm.profit_30d / NULLIF(rm.revenue_30d, 0)) * 100.0,
- rm.sales_30d / NULLIF(rm.total_avg_stock_units_30d, 0) -- Simple unit-based turnover
- FROM public.categories c -- Start from categories to include those with no products yet
- LEFT JOIN RollupMetrics rm ON c.cat_id = rm.cat_id
+ -- Rolled-up metrics (total including children)
+ cm.product_count,
+ cm.active_product_count,
+ cm.replenishable_product_count,
+ cm.current_stock_units,
+ cm.current_stock_cost,
+ cm.current_stock_retail,
+ cm.sales_7d, cm.revenue_7d,
+ cm.sales_30d, cm.revenue_30d, cm.profit_30d, cm.cogs_30d,
+ cm.sales_365d, cm.revenue_365d,
+ cm.lifetime_sales, cm.lifetime_revenue,
+ -- Direct metrics (just this category)
+ cm.direct_product_count,
+ cm.direct_active_product_count,
+ cm.direct_replenishable_product_count,
+ cm.direct_current_stock_units,
+ cm.direct_current_stock_cost,
+ cm.direct_current_stock_retail,
+ cm.direct_sales_7d, cm.direct_revenue_7d,
+ cm.direct_sales_30d, cm.direct_revenue_30d, cm.direct_profit_30d, cm.direct_cogs_30d,
+ cm.direct_sales_365d, cm.direct_revenue_365d,
+ cm.direct_lifetime_sales, cm.direct_lifetime_revenue,
+ -- KPIs - Calculate margin only for categories with significant revenue
+ CASE
+ WHEN cm.revenue_30d >= _min_revenue THEN
+ ((cm.revenue_30d - cm.cogs_30d) / cm.revenue_30d) * 100.0
+ ELSE NULL -- No margin for low/no revenue categories
+ END,
+ -- Stock Turn calculation
+ CASE
+ WHEN cm.total_avg_stock_units_30d > 0 THEN
+ cm.sales_30d / cm.total_avg_stock_units_30d
+ ELSE NULL -- No stock turn if no average stock
+ END
+ FROM CombinedMetrics cm
ON CONFLICT (category_id) DO UPDATE SET
category_name = EXCLUDED.category_name,
category_type = EXCLUDED.category_type,
parent_id = EXCLUDED.parent_id,
last_calculated = EXCLUDED.last_calculated,
+ -- Update rolled-up metrics
product_count = EXCLUDED.product_count,
active_product_count = EXCLUDED.active_product_count,
replenishable_product_count = EXCLUDED.replenishable_product_count,
@@ -172,6 +250,19 @@ BEGIN
profit_30d = EXCLUDED.profit_30d, cogs_30d = EXCLUDED.cogs_30d,
sales_365d = EXCLUDED.sales_365d, revenue_365d = EXCLUDED.revenue_365d,
lifetime_sales = EXCLUDED.lifetime_sales, lifetime_revenue = EXCLUDED.lifetime_revenue,
+ -- Update direct metrics
+ direct_product_count = EXCLUDED.direct_product_count,
+ direct_active_product_count = EXCLUDED.direct_active_product_count,
+ direct_replenishable_product_count = EXCLUDED.direct_replenishable_product_count,
+ direct_current_stock_units = EXCLUDED.direct_current_stock_units,
+ direct_stock_cost = EXCLUDED.direct_stock_cost,
+ direct_stock_retail = EXCLUDED.direct_stock_retail,
+ direct_sales_7d = EXCLUDED.direct_sales_7d, direct_revenue_7d = EXCLUDED.direct_revenue_7d,
+ direct_sales_30d = EXCLUDED.direct_sales_30d, direct_revenue_30d = EXCLUDED.direct_revenue_30d,
+ direct_profit_30d = EXCLUDED.direct_profit_30d, direct_cogs_30d = EXCLUDED.direct_cogs_30d,
+ direct_sales_365d = EXCLUDED.direct_sales_365d, direct_revenue_365d = EXCLUDED.direct_revenue_365d,
+ direct_lifetime_sales = EXCLUDED.direct_lifetime_sales, direct_lifetime_revenue = EXCLUDED.direct_lifetime_revenue,
+ -- Update KPIs
avg_margin_30d = EXCLUDED.avg_margin_30d,
stock_turn_30d = EXCLUDED.stock_turn_30d;
diff --git a/inventory/src/pages/Categories.tsx b/inventory/src/pages/Categories.tsx
index a399662..8a2ceb6 100644
--- a/inventory/src/pages/Categories.tsx
+++ b/inventory/src/pages/Categories.tsx
@@ -106,6 +106,11 @@ interface CategoryMetric {
lifetimeRevenue: string | number;
avgMargin_30d: string | number | null;
stockTurn_30d: string | number | null;
+ direct_active_product_count: number;
+ direct_current_stock_units: number;
+ direct_stock_cost: string | number;
+ direct_revenue_30d: string | number;
+ direct_profit_30d: string | number;
}
// Define response type to avoid type errors
@@ -455,7 +460,84 @@ export function Categories() {
// Build the hierarchical tree structure
const hierarchicalCategories = useMemo(() => {
if (!categories || categories.length === 0) return [];
-
+
+ // DIRECT CALCULATION: Create a map to directly calculate accurate totals
+ // This approach doesn't rely on the tree structure for calculations
+ const directTotalsMap = new Map();
+
+ // First, identify all parent-child relationships
+ const childToParentMap = new Map();
+ categories.forEach(cat => {
+ if (cat.parent_id) {
+ childToParentMap.set(cat.category_id, cat.parent_id);
+ }
+ });
+
+ // Build sets of all descendants for each category
+ const allDescendantsMap = new Map>();
+
+ // Helper to get all descendants recursively
+ const getAllDescendants = (categoryId: string | number): Set => {
+ if (allDescendantsMap.has(categoryId)) {
+ return allDescendantsMap.get(categoryId)!;
+ }
+
+ const descendants = new Set();
+
+ // Find direct children
+ categories.forEach(cat => {
+ if (cat.parent_id === categoryId) {
+ descendants.add(cat.category_id);
+ // Add their descendants recursively
+ const childDescendants = getAllDescendants(cat.category_id);
+ childDescendants.forEach(id => descendants.add(id));
+ }
+ });
+
+ // Cache and return
+ allDescendantsMap.set(categoryId, descendants);
+ return descendants;
+ };
+
+ // Calculate descendants for all categories
+ categories.forEach(cat => {
+ if (!allDescendantsMap.has(cat.category_id)) {
+ getAllDescendants(cat.category_id);
+ }
+ });
+
+ // Now use pre-calculated database values instead of calculating totals directly
+ categories.forEach(cat => {
+ // Use the pre-calculated values from the database
+ const totalRevenue = parseFloat(cat.revenue_30d?.toString() || '0');
+ const totalProfit = parseFloat(cat.profit_30d?.toString() || '0');
+ const totalActiveProducts = cat.active_product_count || 0;
+ const totalStockUnits = cat.current_stock_units || 0;
+ const totalStockCost = parseFloat(cat.current_stock_cost?.toString() || '0');
+
+ // Direct values (for display in tooltips)
+ const directRevenue = parseFloat(cat.direct_revenue_30d?.toString() || '0');
+ const directProfit = parseFloat(cat.direct_profit_30d?.toString() || '0');
+ const directActiveProducts = cat.direct_active_product_count || 0;
+ const directStockUnits = cat.direct_current_stock_units || 0;
+ const directStockCost = parseFloat(cat.direct_stock_cost?.toString() || '0');
+
+ // Set the pre-calculated totals
+ directTotalsMap.set(cat.category_id, {
+ revenue30d: totalRevenue,
+ profit30d: totalProfit,
+ activeProductCount: totalActiveProducts,
+ currentStockUnits: totalStockUnits,
+ currentStockCost: totalStockCost
+ });
+ });
+
// First, create a lookup map by category ID
const categoryMap = new Map();
categories.forEach((cat) => {
@@ -476,14 +558,16 @@ export function Categories() {
if (parent) {
parent.children.push(processedCat);
}
+ } else if (cat.parent_id) {
+ // This is an orphaned category
} else {
// This is a root category
rootCategories.push(processedCat);
}
});
-
- // Compute hierarchy levels and aggregate stats
- const computeHierarchyAndStats = (
+
+ // Set hierarchy levels and use the direct aggregates we calculated
+ const computeHierarchyAndLevels = (
categories: CategoryWithChildren[],
level = 0
) => {
@@ -493,60 +577,49 @@ export function Categories() {
cat.isLast = index === arr.length - 1;
cat.isExpanded = expandedCategories.has(cat.category_id);
- // Process children first to ensure we have their aggregated values
+ // Process children to set their hierarchy levels
const children =
cat.children.length > 0
- ? computeHierarchyAndStats(cat.children, level + 1)
+ ? computeHierarchyAndLevels(cat.children, level + 1)
: [];
- // Calculate this category's own direct stats for clarity
- const ownStats = {
- activeProductCount: cat.active_product_count || 0,
- currentStockUnits: cat.current_stock_units || 0,
- currentStockCost: parseFloat(
- cat.current_stock_cost?.toString() || "0"
- ),
- revenue30d: parseFloat(cat.revenue_30d?.toString() || "0"),
- profit30d: parseFloat(cat.profit_30d?.toString() || "0"),
- avg_margin_30d: parseFloat(cat.avg_margin_30d?.toString() || "0"),
- };
-
- // For leaf nodes (no children), aggregated stats = own stats
- if (children.length === 0) {
- cat.aggregatedStats = { ...ownStats };
- return cat;
+ // Make sure we set aggregatedStats for ALL categories, not just those with children
+ // First check if we have pre-calculated values
+ const totals = directTotalsMap.get(cat.category_id);
+
+ if (totals) {
+ // Use our pre-calculated totals for all metrics
+ cat.aggregatedStats = {
+ activeProductCount: totals.activeProductCount,
+ currentStockUnits: totals.currentStockUnits,
+ currentStockCost: totals.currentStockCost,
+ revenue30d: totals.revenue30d,
+ profit30d: totals.profit30d,
+ avg_margin_30d: totals.revenue30d > 0
+ ? (totals.profit30d / totals.revenue30d) * 100
+ : 0
+ };
+ } else {
+ // If we don't have pre-calculated values (shouldn't happen with our algorithm)
+ // Ensure every category still has aggregatedStats set with its direct values
+ cat.aggregatedStats = {
+ activeProductCount: cat.direct_active_product_count || 0,
+ currentStockUnits: cat.direct_current_stock_units || 0,
+ currentStockCost: parseFloat(cat.direct_stock_cost?.toString() || "0"),
+ revenue30d: parseFloat(cat.direct_revenue_30d?.toString() || "0"),
+ profit30d: parseFloat(cat.direct_profit_30d?.toString() || "0"),
+ avg_margin_30d: parseFloat(cat.avg_margin_30d?.toString() || "0")
+ };
}
- // For parents, calculate aggregated stats = own stats + sum of all children's aggregated stats
- const aggregatedStats = { ...ownStats }; // Start with own stats
-
- // Add all children's AGGREGATED stats (not direct stats)
- children.forEach((child) => {
- if (child.aggregatedStats) {
- aggregatedStats.activeProductCount +=
- child.aggregatedStats.activeProductCount;
- aggregatedStats.currentStockUnits +=
- child.aggregatedStats.currentStockUnits;
- aggregatedStats.currentStockCost +=
- child.aggregatedStats.currentStockCost;
- aggregatedStats.revenue30d += child.aggregatedStats.revenue30d;
- aggregatedStats.profit30d += child.aggregatedStats.profit30d;
- }
- });
-
- // Recalculate margin based on total profit and revenue
- if (aggregatedStats.revenue30d > 0) {
- aggregatedStats.avg_margin_30d =
- (aggregatedStats.profit30d / aggregatedStats.revenue30d) * 100;
- }
-
- cat.aggregatedStats = aggregatedStats;
return cat;
});
};
- // Compute hierarchy levels and aggregate stats for all categories
- return computeHierarchyAndStats(rootCategories);
+ // Apply hierarchy levels and use our pre-calculated totals
+ const result = computeHierarchyAndLevels(rootCategories);
+
+ return result;
}, [categories, expandedCategories]);
// Recursive function to render category rows with streamlined stat display
@@ -636,7 +709,7 @@ export function Categories() {
Directly in '{category.category_name}':{" "}
- {formatNumber(category.active_product_count)}
+ {formatNumber(category.direct_active_product_count)}
@@ -662,7 +735,7 @@ export function Categories() {
Directly in '{category.category_name}':{" "}
- {formatNumber(category.current_stock_units)}
+ {formatNumber(category.direct_current_stock_units)}
@@ -687,7 +760,7 @@ export function Categories() {
Directly in '{category.category_name}':{" "}
- {formatCurrency(category.current_stock_cost)}
+ {formatCurrency(category.direct_stock_cost)}
@@ -712,12 +785,12 @@ export function Categories() {
Directly from '{category.category_name}':{" "}
- {formatCurrency(category.revenue_30d)}
+ {formatCurrency(category.direct_revenue_30d)}
) : (
- formatCurrency(category.revenue_30d)
+ formatCurrency(category.aggregatedStats ? category.aggregatedStats.revenue30d : category.revenue_30d)
)}
@@ -737,12 +810,12 @@ export function Categories() {
Directly from '{category.category_name}':{" "}
- {formatCurrency(category.profit_30d)}
+ {formatCurrency(category.direct_profit_30d)}
) : (
- formatCurrency(category.profit_30d)
+ formatCurrency(category.aggregatedStats ? category.aggregatedStats.profit30d : category.profit_30d)
)}