2020-04-20 21:20:26 +00:00
|
|
|
"""books - Tools for loading the books"""
|
|
|
|
# Copyright © 2020 Brett Smith
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Affero General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU Affero General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
import datetime
|
|
|
|
|
2020-04-21 14:47:13 +00:00
|
|
|
from pathlib import Path
|
|
|
|
|
2020-04-21 15:58:28 +00:00
|
|
|
from beancount import loader as bc_loader
|
|
|
|
|
2020-04-20 21:20:26 +00:00
|
|
|
from typing import (
|
2020-04-21 15:58:28 +00:00
|
|
|
Any,
|
2020-04-20 21:20:26 +00:00
|
|
|
Iterable,
|
2020-05-05 18:31:08 +00:00
|
|
|
Iterator,
|
2020-04-21 14:47:13 +00:00
|
|
|
Mapping,
|
2020-04-20 21:20:26 +00:00
|
|
|
NamedTuple,
|
|
|
|
Optional,
|
|
|
|
Union,
|
|
|
|
)
|
2020-04-21 15:58:28 +00:00
|
|
|
from .beancount_types import (
|
2020-06-07 13:04:53 +00:00
|
|
|
Error,
|
|
|
|
Errors,
|
2020-04-21 15:58:28 +00:00
|
|
|
LoadResult,
|
|
|
|
)
|
2020-04-20 21:20:26 +00:00
|
|
|
|
2020-04-21 14:47:13 +00:00
|
|
|
PathLike = Union[str, Path]
|
2020-04-21 13:51:27 +00:00
|
|
|
Year = Union[int, datetime.date]
|
|
|
|
|
2020-04-20 21:20:26 +00:00
|
|
|
class FiscalYear(NamedTuple):
|
|
|
|
month: int = 3
|
|
|
|
day: int = 1
|
|
|
|
|
|
|
|
def for_date(self, date: Optional[datetime.date]=None) -> int:
|
|
|
|
if date is None:
|
|
|
|
date = datetime.date.today()
|
|
|
|
if (date.month, date.day) < self:
|
|
|
|
return date.year - 1
|
|
|
|
else:
|
|
|
|
return date.year
|
|
|
|
|
2020-06-10 19:16:01 +00:00
|
|
|
def first_date(self, year: Year) -> datetime.date:
|
|
|
|
if isinstance(year, datetime.date):
|
|
|
|
year = self.for_date(year)
|
|
|
|
return datetime.date(year, self.month, self.day)
|
|
|
|
|
|
|
|
def last_date(self, year: Year) -> datetime.date:
|
|
|
|
return self.next_fy_date(year) - datetime.timedelta(days=1)
|
|
|
|
|
|
|
|
def next_fy_date(self, year: Year) -> datetime.date:
|
|
|
|
if isinstance(year, datetime.date):
|
|
|
|
year = self.for_date(year)
|
|
|
|
return datetime.date(year + 1, self.month, self.day)
|
|
|
|
|
2020-04-21 13:51:27 +00:00
|
|
|
def range(self, from_fy: Year, to_fy: Optional[Year]=None) -> Iterable[int]:
|
2020-04-20 21:20:26 +00:00
|
|
|
"""Return a range of fiscal years
|
|
|
|
|
|
|
|
Both arguments can be either a year (represented as an integer) or a
|
|
|
|
date. Dates will be converted into a year by calling for_date() on
|
|
|
|
them.
|
|
|
|
|
|
|
|
If the first argument is negative or below 1000, it will be treated as
|
|
|
|
an offset. You'll get a range of fiscal years between the second
|
|
|
|
argument offset by this amount.
|
|
|
|
|
|
|
|
If the second argument is omitted, it defaults to the current fiscal
|
|
|
|
year.
|
|
|
|
|
|
|
|
Note that unlike normal Python ranges, these ranges include the final
|
|
|
|
fiscal year.
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
range(2015) # Iterate all fiscal years from 2015 to today, inclusive
|
|
|
|
|
|
|
|
range(-1) # Iterate the previous fiscal year and current fiscal year
|
|
|
|
"""
|
|
|
|
if not isinstance(from_fy, int):
|
|
|
|
from_fy = self.for_date(from_fy)
|
|
|
|
if to_fy is None:
|
|
|
|
to_fy = self.for_date()
|
|
|
|
elif not isinstance(to_fy, int):
|
|
|
|
to_fy = self.for_date(to_fy - datetime.timedelta(days=1))
|
|
|
|
if from_fy < 1:
|
|
|
|
from_fy += to_fy
|
|
|
|
elif from_fy < 1000:
|
|
|
|
from_fy, to_fy = to_fy, from_fy + to_fy
|
|
|
|
return range(from_fy, to_fy + 1)
|
2020-04-21 14:47:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Loader:
|
|
|
|
"""Load Beancount books organized by fiscal year"""
|
|
|
|
|
|
|
|
def __init__(self,
|
|
|
|
books_root: Path,
|
|
|
|
fiscal_year: FiscalYear,
|
|
|
|
) -> None:
|
|
|
|
"""Set up a books loader
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
* books_root: A Path to a Beancount books checkout.
|
|
|
|
* fiscal_year: A FiscalYear object, used to determine what books to
|
|
|
|
load for a given date range.
|
|
|
|
"""
|
|
|
|
self.books_root = books_root
|
|
|
|
self.fiscal_year = fiscal_year
|
|
|
|
|
2020-05-05 18:31:08 +00:00
|
|
|
def _iter_fy_books(self, fy_range: Iterable[int]) -> Iterator[Path]:
|
|
|
|
for year in fy_range:
|
2020-05-16 14:27:06 +00:00
|
|
|
path = Path(self.books_root, 'books', f'{year}.beancount')
|
2020-05-05 18:31:08 +00:00
|
|
|
if path.exists():
|
|
|
|
yield path
|
2020-04-21 14:47:13 +00:00
|
|
|
|
2020-05-25 14:37:21 +00:00
|
|
|
def _load_paths(self, paths: Iterator[Path]) -> LoadResult:
|
2020-04-21 14:47:13 +00:00
|
|
|
try:
|
2020-05-25 14:37:21 +00:00
|
|
|
entries, errors, options_map = bc_loader.load_file(next(paths))
|
2020-04-21 14:47:13 +00:00
|
|
|
except StopIteration:
|
2020-05-16 14:27:06 +00:00
|
|
|
entries, errors, options_map = [], [], {}
|
2020-05-25 14:37:21 +00:00
|
|
|
for load_path in paths:
|
2020-05-16 14:27:06 +00:00
|
|
|
new_entries, new_errors, new_options = bc_loader.load_file(load_path)
|
|
|
|
# We only want transactions from the new fiscal year.
|
|
|
|
# We don't want the opening balance, duplicate definitions, etc.
|
|
|
|
fy_filename = str(load_path.parent.parent / load_path.name)
|
|
|
|
entries.extend(
|
|
|
|
entry for entry in new_entries
|
|
|
|
if entry.meta.get('filename') == fy_filename
|
|
|
|
)
|
|
|
|
errors.extend(new_errors)
|
|
|
|
return entries, errors, options_map
|
2020-05-25 14:37:21 +00:00
|
|
|
|
2020-06-04 13:03:10 +00:00
|
|
|
def _path_year(self, path: Path) -> int:
|
|
|
|
return int(path.stem)
|
2020-05-25 14:37:21 +00:00
|
|
|
|
2020-06-04 13:03:10 +00:00
|
|
|
def load_all(self, from_year: Optional[Year]=None) -> LoadResult:
|
|
|
|
"""Load all of the books from a starting FY
|
|
|
|
|
|
|
|
This method loads all of the books, starting from the fiscal year you
|
|
|
|
specify.
|
|
|
|
|
|
|
|
* Pass in a date to start from the FY for that date.
|
|
|
|
* Pass in an integer >= 1000 to start from that year.
|
|
|
|
* Pass in a smaller integer to start from an FY relative to today
|
|
|
|
(e.g., -2 starts two FYs before today).
|
|
|
|
* Pass is no argument to load all books from the first available FY.
|
|
|
|
|
|
|
|
This method finds books by globbing the filesystem. It still loads
|
|
|
|
each fiscal year in sequence to provide the best cache utilization.
|
2020-05-25 14:37:21 +00:00
|
|
|
"""
|
|
|
|
path = Path(self.books_root, 'books')
|
|
|
|
fy_paths = list(path.glob('[1-9][0-9][0-9][0-9].beancount'))
|
2020-06-04 13:03:10 +00:00
|
|
|
fy_paths.sort(key=self._path_year)
|
|
|
|
if from_year is not None:
|
|
|
|
if not isinstance(from_year, int):
|
|
|
|
from_year = self.fiscal_year.for_date(from_year)
|
|
|
|
elif from_year < 1000:
|
|
|
|
from_year = self.fiscal_year.for_date() + from_year
|
|
|
|
for index, path in enumerate(fy_paths):
|
|
|
|
if self._path_year(path) >= from_year:
|
|
|
|
fy_paths = fy_paths[index:]
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
fy_paths = []
|
2020-05-25 14:37:21 +00:00
|
|
|
return self._load_paths(iter(fy_paths))
|
|
|
|
|
|
|
|
def load_fy_range(self,
|
|
|
|
from_fy: Year,
|
|
|
|
to_fy: Optional[Year]=None,
|
|
|
|
) -> LoadResult:
|
|
|
|
"""Load books for a range of fiscal years
|
|
|
|
|
|
|
|
This method generates a range of fiscal years by calling
|
2020-06-04 13:03:10 +00:00
|
|
|
FiscalYear.range() with its arguments. It loads all the books within
|
|
|
|
that range.
|
2020-05-25 14:37:21 +00:00
|
|
|
"""
|
|
|
|
fy_range = self.fiscal_year.range(from_fy, to_fy)
|
|
|
|
fy_paths = self._iter_fy_books(fy_range)
|
|
|
|
return self._load_paths(fy_paths)
|
2020-06-07 13:04:53 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def load_none(cls, config_path: Optional[PathLike]=None, lineno: int=0) -> LoadResult:
|
|
|
|
"""Load no books and generate an error about it
|
|
|
|
|
|
|
|
This is a convenience method for reporting tools that already handle
|
|
|
|
general Beancount errors. If a configuration problem prevents them from
|
|
|
|
loading the books, they can call this method in place of a regular
|
|
|
|
loading method, and then continue on their normal code path.
|
|
|
|
|
|
|
|
The path and line number given in the arguments will be named as the
|
|
|
|
source of the error.
|
|
|
|
"""
|
|
|
|
source = {
|
|
|
|
'filename': str(config_path or 'conservancy_beancount.ini'),
|
|
|
|
'lineno': lineno,
|
|
|
|
}
|
|
|
|
errors: Errors = [Error(source, "no books to load in configuration", None)]
|
|
|
|
return [], errors, {}
|