Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat chinese account #74

Open
wants to merge 13 commits into
base: main
Choose a base branch
from
236 changes: 236 additions & 0 deletions .env/Lib/site-packages/beancount/core/account.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
"""Functions that operate on account strings.

These account objects are rather simple and dumb; they do not contain the list
of their associated postings. This is achieved by building a realization; see
realization.py for details.
"""
__copyright__ = "Copyright (C) 2013-2016 Martin Blais"
__license__ = "GNU GPLv2"

import re
import os
from os import path

from beancount.utils import regexp_utils


# Component separator for account names.
# pylint: disable=invalid-name
sep = ':'


# # Regular expression string that matches valid account name components.
# # Categories are:
# # Lu: Uppercase letters.
# # L: All letters.
# # Nd: Decimal numbers.
# ACC_COMP_TYPE_RE = regexp_utils.re_replace_unicode(r"[\p{Lu}][\p{L}\p{Nd}\-]*")
# ACC_COMP_NAME_RE = regexp_utils.re_replace_unicode(r"[\p{Lu}\p{Nd}][\p{L}\p{Nd}\-]*")

# Regular expression string that matches valid account name components.
# Categories are:
# Lu: Uppercase letters.
# Lu1: Uppercase letters And CJK letters
# L: All letters.
# Nd: Decimal numbers.
ACC_COMP_TYPE_RE = regexp_utils.re_replace_unicode(r"[\p{Lu}][\p{L}\p{Nd}\-]*")
ACC_COMP_NAME_RE = regexp_utils.re_replace_unicode(r"[\p{Lu1}\p{Nd}][\p{L}\p{Nd}\-]*")

# Regular expression string that matches a valid account. {5672c7270e1e}
ACCOUNT_RE = "(?:{})(?:{}{})+".format(ACC_COMP_TYPE_RE, sep, ACC_COMP_NAME_RE)


# A dummy object which stands for the account type. Values in custom directives
# use this to disambiguate between string objects and account names.
TYPE = '<AccountDummy>'


def is_valid(string):
"""Return true if the given string is a valid account name.
This does not check for the root account types, just the general syntax.

Args:
string: A string, to be checked for account name pattern.
Returns:
A boolean, true if the string has the form of an account's name.
"""
return (isinstance(string, str) and
bool(re.match('{}$'.format(ACCOUNT_RE), string)))


def join(*components):
"""Join the names with the account separator.

Args:
*components: Strings, the components of an account name.
Returns:
A string, joined in a single account name.
"""
return sep.join(components)


def split(account_name):
"""Split an account's name into its components.

Args:
account_name: A string, an account name.
Returns:
A list of strings, the components of the account name (without the separators).
"""
return account_name.split(sep)


def parent(account_name):
"""Return the name of the parent account of the given account.

Args:
account_name: A string, the name of the account whose parent to return.
Returns:
A string, the name of the parent account of this account.
"""
assert isinstance(account_name, str), account_name
if not account_name:
return None
components = account_name.split(sep)
components.pop(-1)
return sep.join(components)


def leaf(account_name):
"""Get the name of the leaf of this account.

Args:
account_name: A string, the name of the account whose leaf name to return.
Returns:
A string, the name of the leaf of the account.
"""
assert isinstance(account_name, str)
return account_name.split(sep)[-1] if account_name else None


def sans_root(account_name):
"""Get the name of the account without the root.

For example, an input of 'Assets:BofA:Checking' will produce 'BofA:Checking'.

Args:
account_name: A string, the name of the account whose leaf name to return.
Returns:
A string, the name of the non-root portion of this account name.
"""
assert isinstance(account_name, str)
components = account_name.split(sep)[1:]
return join(*components) if account_name else None


def root(num_components, account_name):
"""Return the first few components of an account's name.

Args:
num_components: An integer, the number of components to return.
account_name: A string, an account name.
Returns:
A string, the account root up to 'num_components' components.
"""
return join(*(split(account_name)[:num_components]))


def has_component(account_name, component):
"""Return true if one of the account contains a given component.

Args:
account_name: A string, an account name.
component: A string, a component of an account name. For instance,
``Food`` in ``Expenses:Food:Restaurant``. All components are considered.
Returns:
Boolean: true if the component is in the account. Note that a component
name must be whole, that is ``NY`` is not in ``Expenses:Taxes:StateNY``.
"""
return bool(re.search('(^|:){}(:|$)'.format(component), account_name))


def commonprefix(accounts):
"""Return the common prefix of a list of account names.

Args:
accounts: A sequence of account name strings.
Returns:
A string, the common parent account. If none, returns an empty string.
"""
accounts_lists = [account_.split(sep)
for account_ in accounts]
# Note: the os.path.commonprefix() function just happens to work here.
# Inspect its code, and even the special case of no common prefix
# works well with str.join() below.
common_list = path.commonprefix(accounts_lists)
return sep.join(common_list)


def walk(root_directory):
"""A version of os.walk() which yields directories that are valid account names.

This only yields directories that are accounts... it skips the other ones.
For convenience, it also yields you the account's name.

Args:
root_directory: A string, the name of the root of the hierarchy to be walked.
Yields:
Tuples of (root, account-name, dirs, files), similar to os.walk().
"""
for root, dirs, files in os.walk(root_directory):
dirs.sort()
files.sort()
relroot = root[len(root_directory)+1:]
account_name = relroot.replace(os.sep, sep)
if is_valid(account_name):
yield (root, account_name, dirs, files)


def parent_matcher(account_name):
"""Build a predicate that returns whether an account is under the given one.

Args:
account_name: The name of the parent account we want to check for.
Returns:
A callable, which, when called, will return true if the given account is a
child of ``account_name``.
"""
return re.compile(r'{}($|{})'.format(re.escape(account_name), sep)).match


def parents(account_name):
"""A generator of the names of the parents of this account, including this account.

Args:
account_name: The name of the account we want to start iterating from.
Returns:
A generator of account name strings.
"""
while account_name:
yield account_name
account_name = parent(account_name)


class AccountTransformer:
"""Account name transformer.

This is used to support Win... huh, filesystems and platforms which do not
support colon characters.

Attributes:
rsep: A character string, the new separator to use in link names.
"""
def __init__(self, rsep=None):
self.rsep = rsep

def render(self, account_name):
"Convert the account name to a transformed account name."
return (account_name
if self.rsep is None
else account_name.replace(sep, self.rsep))

def parse(self, transformed_name):
"Convert the transform account name to an account name."
return (transformed_name
if self.rsep is None
else transformed_name.replace(self.rsep, sep))
Loading