- Fix pipeline granularity: add market_year to cleaned/serving SQL models - Add DuckDB data access layer with async query functions (analytics.py) - Build Chart.js dashboard: supply/demand, STU ratio, top producers, YoY table - Add country comparison page with multi-select picker - Replace items CRUD with read-only commodity API (list, metrics, countries, CSV) - Configure BeanFlows plan tiers (Free/Starter/Pro) with feature gating - Rewrite public pages for coffee market intelligence positioning - Remove boilerplate items schema, update health check for DuckDB - Add test suite: 139 tests passing (dashboard, API, billing) Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
107 lines
3.1 KiB
SQL
107 lines
3.1 KiB
SQL
MODEL (
|
|
name serving.commodity_metrics,
|
|
kind INCREMENTAL_BY_TIME_RANGE (
|
|
time_column ingest_date
|
|
),
|
|
start '2006-08-01',
|
|
cron '@daily'
|
|
);
|
|
|
|
-- CTE to calculate country-level derived metrics
|
|
WITH country_metrics AS (
|
|
SELECT
|
|
commodity_code,
|
|
commodity_name,
|
|
country_code,
|
|
country_name,
|
|
market_year,
|
|
ingest_date,
|
|
Production,
|
|
Imports,
|
|
Exports,
|
|
Total_Distribution,
|
|
Ending_Stocks,
|
|
-- Derived metrics per country, mirroring Python script
|
|
(Production + Imports - Exports) AS Net_Supply,
|
|
(Exports - Imports) AS Trade_Balance,
|
|
(Production + Imports - Exports) - Total_Distribution AS Supply_Demand_Balance,
|
|
-- Handle division by zero for Stock-to-Use Ratio
|
|
(Ending_Stocks / NULLIF(Total_Distribution, 0)) * 100 AS Stock_to_Use_Ratio_pct,
|
|
-- Calculate Production YoY percentage change using a window function
|
|
(Production - LAG(Production, 1, 0) OVER (PARTITION BY commodity_code, country_code ORDER BY market_year, ingest_date)) / NULLIF(LAG(Production, 1, 0) OVER (PARTITION BY commodity_code, country_code ORDER BY market_year, ingest_date), 0) * 100 AS Production_YoY_pct
|
|
FROM cleaned.psdalldata__commodity_pivoted
|
|
),
|
|
global_aggregates AS (
|
|
SELECT
|
|
commodity_code,
|
|
commodity_name,
|
|
NULL::TEXT AS country_code, -- Use NULL for global aggregates
|
|
'Global' AS country_name,
|
|
market_year,
|
|
ingest_date,
|
|
SUM(Production) AS Production,
|
|
SUM(Imports) AS Imports,
|
|
SUM(Exports) AS Exports,
|
|
SUM(Total_Distribution) AS Total_Distribution,
|
|
SUM(Ending_Stocks) AS Ending_Stocks
|
|
FROM cleaned.psdalldata__commodity_pivoted
|
|
GROUP BY
|
|
commodity_code,
|
|
commodity_name,
|
|
market_year,
|
|
ingest_date
|
|
),
|
|
-- CTE to calculate derived metrics for global aggregates
|
|
global_metrics AS (
|
|
SELECT
|
|
commodity_code,
|
|
commodity_name,
|
|
country_code,
|
|
country_name,
|
|
market_year,
|
|
ingest_date,
|
|
Production,
|
|
Imports,
|
|
Exports,
|
|
Total_Distribution,
|
|
Ending_Stocks,
|
|
(Production + Imports - Exports) AS Net_Supply,
|
|
(Exports - Imports) AS Trade_Balance,
|
|
(Production + Imports - Exports) - Total_Distribution AS Supply_Demand_Balance,
|
|
(Ending_Stocks / NULLIF(Total_Distribution, 0)) * 100 AS Stock_to_Use_Ratio_pct,
|
|
(Production - LAG(Production, 1, 0) OVER (PARTITION BY commodity_code ORDER BY market_year, ingest_date)) / NULLIF(LAG(Production, 1, 0) OVER (PARTITION BY commodity_code ORDER BY market_year, ingest_date), 0) * 100 AS Production_YoY_pct
|
|
FROM global_aggregates
|
|
)
|
|
-- Combine country-level and global-level data into a single output
|
|
SELECT
|
|
commodity_code,
|
|
commodity_name,
|
|
country_code,
|
|
country_name,
|
|
market_year,
|
|
ingest_date,
|
|
Production,
|
|
Imports,
|
|
Exports,
|
|
Total_Distribution,
|
|
Ending_Stocks,
|
|
Net_Supply,
|
|
Trade_Balance,
|
|
Supply_Demand_Balance,
|
|
Stock_to_Use_Ratio_pct,
|
|
Production_YoY_pct
|
|
FROM (
|
|
SELECT
|
|
*
|
|
FROM country_metrics
|
|
UNION ALL
|
|
SELECT
|
|
*
|
|
FROM global_metrics
|
|
) AS combined_data
|
|
ORDER BY
|
|
commodity_name,
|
|
country_name,
|
|
market_year,
|
|
ingest_date;
|