I need to express "fiscal {month,quarter,year}" compactly, e.g. in a database column or HTTP endpoint. I've done some literature research. Looker uses fiscal_month_offset, which seems as good a standard as any. But there's nothing on storage/transfer formats. I'm considering these:
# [u-...] inspired by Temporal API
2024[u-fmo=-7] = 2023-06-01 - 2024-05-31
2025Q2[u-fmo=+3] = 2025-07-01 - 2026-09-30
2040M10[u-fmo=-5] = 2040-05-01 - 2040-05-31
# I picked ~ somewhat arbitrarily as a delimiter
FY2024~-7 = 2023-06-01 - 2024-05-31
FY2025Q2~+3 = 2025-07-01 - 2026-09-30
2040M10~-5 = 2040-05-01 - 2040-05-31
# The {...} sets off the month offset visually
2024{-7} = 2023-06-01 - 2024-05-31
2025Q2{3} = 2025-07-01 - 2026-09-30
2040M10{-5} = 2040-05-01 - 2040-05-31
Does anyone have a favorite? Another idea?
I'm tempted to say: just store the start and end dates in your database, and use code to convert to and from the fiscal year format.
I certainly could do that. One key downside is size.
I can store fiscal_year
and fiscal_offset
in two smallint
s for 4 bytes. I can store fiscal_year
, fiscal_year_start
, and fiscal_year_end
in one smallint
and two Date
s for 10 bytes. On a table with 100,000,000 records, that's an extra 600MB at minimum -- more, depending on how well I play column Tetris. (I could omit fiscal_year
, but by not storing it, I can't partition on it.)
I'm going to have to convert one way or the other in queries that need to compare fiscal and calendar periods.
If I store (fiscal_year, fiscal_month_offset)
, I need to convert to query by calendar year
select *
from events
where fiscal_year >= calendar_year_to_fiscal(?, ?)
and fiscal_year <= calendar_year_to_fiscal(?, ?);
Whereas if I store (calendar_start, calendar_end)
, I need to convert to query by fiscal year:
select *
from events
where calendar_start >= fiscal_year_start(?, ?)
and calendar_end <= fiscal_year_end(?, ?);
If regular dates will make your life easier, and if the only downside of two dates is size, I would say double check if that's really a downside at all:
100M rows really something you're going to have?
Probably within a few months. Definitely within 12 months.
DATE takes 3 bytes vs. the 5 bytes of DATETIME
On Postgres, timestamp is 8B, timestamp with time zone is 8B, and date is 4B.
I still don't think saving 300MB is going to be your main concern on a 100GB database :-)
Fair. Thank you for taking the time to make a suggestion. I appreciate the assist.
Last updated: Apr 03 2025 at 23:38 UTC