historical: Add Beancount output format.
This commit is contained in:
parent
7c11ae408c
commit
3a3afb7978
3 changed files with 57 additions and 14 deletions
|
@ -172,9 +172,32 @@ class LedgerFormatter(Formatter):
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class BeancountFormatter(LedgerFormatter):
|
||||||
|
COST_FMT = '{{{}}}'
|
||||||
|
|
||||||
|
def __init__(self, cost_rates, price_rates=None,
|
||||||
|
signed_currencies=(), base_fmt='###0.###',
|
||||||
|
rate_precision=5, denomination=None):
|
||||||
|
super().__init__(
|
||||||
|
cost_rates,
|
||||||
|
price_rates,
|
||||||
|
(),
|
||||||
|
base_fmt.replace(',', ''),
|
||||||
|
rate_precision,
|
||||||
|
denomination,
|
||||||
|
)
|
||||||
|
|
||||||
|
def price_rate(self, from_amt, from_curr, to_curr):
|
||||||
|
if self.price_rates is None:
|
||||||
|
return None
|
||||||
|
else:
|
||||||
|
return self.price_rates.convert(from_amt, from_curr, to_curr)
|
||||||
|
|
||||||
|
|
||||||
class Formats(enum.Enum):
|
class Formats(enum.Enum):
|
||||||
RAW = Formatter
|
RAW = Formatter
|
||||||
LEDGER = LedgerFormatter
|
LEDGER = LedgerFormatter
|
||||||
|
BEANCOUNT = BeancountFormatter
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_arg(cls, s):
|
def from_arg(cls, s):
|
||||||
|
|
|
@ -110,8 +110,9 @@ class Configuration:
|
||||||
hist_parser.add_argument(
|
hist_parser.add_argument(
|
||||||
'--output-format',
|
'--output-format',
|
||||||
type=historical.Formats.from_arg,
|
type=historical.Formats.from_arg,
|
||||||
choices=[fmt.name.lower() for fmt in historical.Formats],
|
help="Output format."
|
||||||
help="Output format. Choices are %(choices)s. Default `raw`.",
|
" Choices are `raw`, `ledger`, `beancount`."
|
||||||
|
" Default `raw`.",
|
||||||
)
|
)
|
||||||
# --ledger and --no-ledger predate --output-format.
|
# --ledger and --no-ledger predate --output-format.
|
||||||
hist_parser.add_argument(
|
hist_parser.add_argument(
|
||||||
|
|
|
@ -40,6 +40,7 @@ class FakeConfig:
|
||||||
output = pytest.fixture(lambda: io.StringIO())
|
output = pytest.fixture(lambda: io.StringIO())
|
||||||
parametrize_format = pytest.mark.parametrize('output_format', [
|
parametrize_format = pytest.mark.parametrize('output_format', [
|
||||||
oxrhist.Formats.LEDGER,
|
oxrhist.Formats.LEDGER,
|
||||||
|
oxrhist.Formats.BEANCOUNT,
|
||||||
])
|
])
|
||||||
|
|
||||||
@pytest.fixture(scope='module')
|
@pytest.fixture(scope='module')
|
||||||
|
@ -85,20 +86,38 @@ def lines_from_run(config, output):
|
||||||
def check_fx_amount(config, lines, amount, cost, fx_code, fx_sign=None, price=None):
|
def check_fx_amount(config, lines, amount, cost, fx_code, fx_sign=None, price=None):
|
||||||
if price is None:
|
if price is None:
|
||||||
price = cost
|
price = cost
|
||||||
|
rate_fmt = f'{{}} {re.escape(fx_code)}'
|
||||||
cost = re.escape(cost) + r'\d*'
|
cost = re.escape(cost) + r'\d*'
|
||||||
price = re.escape(price) + r'\d*'
|
price = re.escape(price) + r'\d*'
|
||||||
if fx_sign is not None and fx_code in config.args.signed_currencies:
|
if config.args.output_format is oxrhist.Formats.LEDGER:
|
||||||
rate_fmt = f'{re.escape(fx_sign)}{{}}'
|
if fx_sign is not None and fx_code in config.args.signed_currencies:
|
||||||
|
rate_fmt = f'{re.escape(fx_sign)}{{}}'
|
||||||
|
cost_re = '{{={}}}'.format(rate_fmt.format(cost))
|
||||||
|
price_re = ' @ {}'.format(rate_fmt.format(price))
|
||||||
else:
|
else:
|
||||||
rate_fmt = f'{{}} {re.escape(fx_code)}'
|
amount = amount.replace(',', '')
|
||||||
pattern = r'^{} {{={}}} @ {}$'.format(
|
cost_re = '{{{}}}'.format(rate_fmt.format(cost))
|
||||||
re.escape(amount),
|
if config.args.from_date is None:
|
||||||
rate_fmt.format(cost),
|
price_re = ''
|
||||||
rate_fmt.format(price),
|
else:
|
||||||
)
|
price_re = ' @ {}'.format(rate_fmt.format(price))
|
||||||
|
pattern = r'^{} {}{}$'.format(re.escape(amount), cost_re, price_re)
|
||||||
line = next(lines, "<EOF>")
|
line = next(lines, "<EOF>")
|
||||||
assert re.match(pattern, line)
|
assert re.match(pattern, line)
|
||||||
|
|
||||||
|
def check_nonfx_amount(config, lines, amount, code=None, sign=None):
|
||||||
|
if config.args.output_format is oxrhist.Formats.LEDGER:
|
||||||
|
if code is None:
|
||||||
|
code = 'USD'
|
||||||
|
sign = '$'
|
||||||
|
if code in config.args.signed_currencies and sign is not None:
|
||||||
|
expected = f'{sign}{amount}\n'
|
||||||
|
else:
|
||||||
|
expected = f'{amount} {code}\n'
|
||||||
|
else:
|
||||||
|
expected = f'{amount.replace(",", "")} {code or "USD"}\n'
|
||||||
|
assert next(lines, "<EOF>") == expected
|
||||||
|
|
||||||
def test_rate_list(single_responder, output, any_date):
|
def test_rate_list(single_responder, output, any_date):
|
||||||
config = build_config(single_responder, any_date)
|
config = build_config(single_responder, any_date)
|
||||||
lines = lines_from_run(config, output)
|
lines = lines_from_run(config, output)
|
||||||
|
@ -144,7 +163,7 @@ def test_ledger_conversion(single_responder, output, any_date, output_format):
|
||||||
amount=300, output_format=output_format)
|
amount=300, output_format=output_format)
|
||||||
lines = lines_from_run(config, output)
|
lines = lines_from_run(config, output)
|
||||||
check_fx_amount(config, lines, '300 ALL', '0.00691', 'USD', '$')
|
check_fx_amount(config, lines, '300 ALL', '0.00691', 'USD', '$')
|
||||||
assert next(lines) == '$2.08\n'
|
check_nonfx_amount(config, lines, '2.08')
|
||||||
assert next(lines, None) is None
|
assert next(lines, None) is None
|
||||||
|
|
||||||
@parametrize_format
|
@parametrize_format
|
||||||
|
@ -173,7 +192,7 @@ def test_redundant_denomination(single_responder, output, any_date, output_forma
|
||||||
output_format=output_format, denomination='USD')
|
output_format=output_format, denomination='USD')
|
||||||
lines = lines_from_run(config, output)
|
lines = lines_from_run(config, output)
|
||||||
check_fx_amount(config, lines, '10.00 ANG', '0.558', 'USD', '$')
|
check_fx_amount(config, lines, '10.00 ANG', '0.558', 'USD', '$')
|
||||||
assert next(lines) == '$5.59\n'
|
check_nonfx_amount(config, lines, '5.59')
|
||||||
assert next(lines, None) is None
|
assert next(lines, None) is None
|
||||||
|
|
||||||
@parametrize_format
|
@parametrize_format
|
||||||
|
@ -182,7 +201,7 @@ def test_from_denomination(single_responder, output, any_date, output_format):
|
||||||
from_currency='USD', to_currency='ALL', amount=10,
|
from_currency='USD', to_currency='ALL', amount=10,
|
||||||
output_format=output_format, denomination='USD')
|
output_format=output_format, denomination='USD')
|
||||||
lines = lines_from_run(config, output)
|
lines = lines_from_run(config, output)
|
||||||
assert next(lines) == '$10.00\n'
|
check_nonfx_amount(config, lines, '10.00')
|
||||||
check_fx_amount(config, lines, '1,445 ALL', '0.00691', 'USD', '$')
|
check_fx_amount(config, lines, '1,445 ALL', '0.00691', 'USD', '$')
|
||||||
assert next(lines, None) is None
|
assert next(lines, None) is None
|
||||||
|
|
||||||
|
@ -197,7 +216,7 @@ def test_rate_precision_added_as_needed(single_responder, output, any_date, outp
|
||||||
# Make sure the rate is specified with enough precision to get the
|
# Make sure the rate is specified with enough precision to get the
|
||||||
# correct conversion amount.
|
# correct conversion amount.
|
||||||
check_fx_amount(config, lines, '63,805.00 RUB', '0.0175204', 'USD', '$')
|
check_fx_amount(config, lines, '63,805.00 RUB', '0.0175204', 'USD', '$')
|
||||||
assert next(lines) == '$1,117.89\n'
|
check_nonfx_amount(config, lines, '1,117.89')
|
||||||
assert next(lines, None) is None
|
assert next(lines, None) is None
|
||||||
|
|
||||||
@parametrize_format
|
@parametrize_format
|
||||||
|
|
Loading…
Reference in a new issue