diff --git a/CHANGES b/CHANGES index 77c78ac..8532c9c 100644 --- a/CHANGES +++ b/CHANGES @@ -1,5 +1,10 @@ Changelog ========= +v0.6 (2024-12-27) +----------------- + +Upgrade to beancount 3 and beangulp. + v0.5 (2024-01-21) ----------------- diff --git a/setup.cfg b/setup.cfg index 0af1f60..495d397 100644 --- a/setup.cfg +++ b/setup.cfg @@ -33,7 +33,8 @@ packages = find: setup_requires = setuptools_scm install_requires = - beancount>=2.3.5,<3.0.0 + beancount>=3 + beangulp scikit-learn>=1.0 numpy>=1.18.0 diff --git a/smart_importer/detector.py b/smart_importer/detector.py index 42b578e..937d650 100644 --- a/smart_importer/detector.py +++ b/smart_importer/detector.py @@ -2,7 +2,7 @@ import logging -from beancount.ingest import similar +from beangulp import similar from smart_importer.hooks import ImporterHook diff --git a/smart_importer/hooks.py b/smart_importer/hooks.py index b767824..e0aaed8 100644 --- a/smart_importer/hooks.py +++ b/smart_importer/hooks.py @@ -36,15 +36,13 @@ def apply_hooks(importer, hooks): unpatched_extract = importer.extract @wraps(unpatched_extract) - def patched_extract_method(file, existing_entries=None): + def patched_extract_method(filepath, existing=None): logger.debug("Calling the importer's extract method.") - imported_entries = unpatched_extract( - file, existing_entries=existing_entries - ) + imported_entries = unpatched_extract(filepath, existing=existing) for hook in hooks: imported_entries = hook( - importer, file, imported_entries, existing_entries + importer, filepath, imported_entries, existing ) return imported_entries diff --git a/smart_importer/predictor.py b/smart_importer/predictor.py index 71593a5..eeec34b 100644 --- a/smart_importer/predictor.py +++ b/smart_importer/predictor.py @@ -83,7 +83,7 @@ def __call__(self, importer, file, imported_entries, existing_entries): A list of entries, modified by this predictor. """ logging.debug("Running %s for file %s", self.__class__.__name__, file) - self.account = importer.file_account(file) + self.account = importer.account(file) self.load_training_data(existing_entries) with self.lock: self.define_pipeline() diff --git a/tests/data_test.py b/tests/data_test.py index 577ea71..d30e21d 100644 --- a/tests/data_test.py +++ b/tests/data_test.py @@ -7,8 +7,8 @@ import pytest from beancount.core.compare import stable_hash_namedtuple -from beancount.ingest.importer import ImporterProtocol from beancount.parser import parser +from beangulp import Importer from smart_importer import PredictPostings, apply_hooks @@ -51,14 +51,14 @@ def test_testset(testset, string_tokenizer): # pylint: disable=unbalanced-tuple-unpacking imported, training_data, expected = _load_testset(testset) - class DummyImporter(ImporterProtocol): - def extract(self, file, existing_entries=None): + class DummyImporter(Importer): + def extract(self, filepath, existing=None): return imported importer = DummyImporter() apply_hooks(importer, [PredictPostings(string_tokenizer=string_tokenizer)]) imported_transactions = importer.extract( - "dummy-data", existing_entries=training_data + "dummy-data", existing=training_data ) for txn1, txn2 in zip(imported_transactions, expected): diff --git a/tests/predictors_test.py b/tests/predictors_test.py index 3460f6f..229edd3 100644 --- a/tests/predictors_test.py +++ b/tests/predictors_test.py @@ -1,8 +1,8 @@ """Tests for the `PredictPayees` and the `PredictPostings` decorator""" # pylint: disable=missing-docstring -from beancount.ingest.importer import ImporterProtocol from beancount.parser import parser +from beangulp import Importer from smart_importer import PredictPayees, PredictPostings from smart_importer.hooks import apply_hooks @@ -132,16 +132,16 @@ DENYLISTED_ACCOUNTS = ["Expenses:Denylisted"] -class BasicTestImporter(ImporterProtocol): - def extract(self, file, existing_entries=None): - if file == "dummy-data": +class BasicTestImporter(Importer): + def extract(self, filepath, existing=None): + if filepath == "dummy-data": return TEST_DATA - if file == "empty": + if filepath == "empty": return [] assert False return [] - def file_account(self, file): + def account(self, filepath): return "Assets:US:BofA:Checking" @@ -166,8 +166,8 @@ def test_no_transactions(): """ POSTING_IMPORTER.extract("empty") PAYEE_IMPORTER.extract("empty") - POSTING_IMPORTER.extract("empty", existing_entries=TRAINING_DATA) - PAYEE_IMPORTER.extract("empty", existing_entries=TRAINING_DATA) + POSTING_IMPORTER.extract("empty", existing=TRAINING_DATA) + PAYEE_IMPORTER.extract("empty", existing=TRAINING_DATA) def test_unchanged_narrations(): @@ -178,7 +178,7 @@ def test_unchanged_narrations(): extracted_narrations = [ transaction.narration for transaction in PAYEE_IMPORTER.extract( - "dummy-data", existing_entries=TRAINING_DATA + "dummy-data", existing=TRAINING_DATA ) ] assert extracted_narrations == correct_narrations @@ -194,7 +194,7 @@ def test_unchanged_first_posting(): extracted_first_postings = [ transaction.postings[0] for transaction in PAYEE_IMPORTER.extract( - "dummy-data", existing_entries=TRAINING_DATA + "dummy-data", existing=TRAINING_DATA ) ] assert extracted_first_postings == correct_first_postings @@ -204,9 +204,7 @@ def test_payee_predictions(): """ Verifies that the decorator adds predicted postings. """ - transactions = PAYEE_IMPORTER.extract( - "dummy-data", existing_entries=TRAINING_DATA - ) + transactions = PAYEE_IMPORTER.extract("dummy-data", existing=TRAINING_DATA) predicted_payees = [transaction.payee for transaction in transactions] assert predicted_payees == PAYEE_PREDICTIONS @@ -218,7 +216,7 @@ def test_account_predictions(): predicted_accounts = [ entry.postings[-1].account for entry in POSTING_IMPORTER.extract( - "dummy-data", existing_entries=TRAINING_DATA + "dummy-data", existing=TRAINING_DATA ) ] assert predicted_accounts == ACCOUNT_PREDICTIONS