activity.models.invoice_reference ================================= .. py:module:: activity.models.invoice_reference Attributes ---------- .. autoapisummary:: activity.models.invoice_reference.KNOWN_SCHEMAS activity.models.invoice_reference.INVALID_REFERENCE_CHARS_EX activity.models.invoice_reference.REFERENCE_EX Classes ------- .. autoapisummary:: activity.models.invoice_reference.InvoiceReference activity.models.invoice_reference.Schema activity.models.invoice_reference.FeriennetSchema activity.models.invoice_reference.ESRSchema activity.models.invoice_reference.RaiffeisenSchema Module Contents --------------- .. py:data:: KNOWN_SCHEMAS :type: dict[str, type[Schema]] .. py:data:: INVALID_REFERENCE_CHARS_EX .. py:data:: REFERENCE_EX .. py:class:: InvoiceReference Bases: :py:obj:`onegov.core.orm.Base`, :py:obj:`onegov.core.orm.mixins.TimestampMixin` A reference pointing to an invoice. References are keys which are used outside the application. Usually a code used on an invoice to enter through online-banking. Each invoice may have multiple references pointing to it. Each refernce is however unique. There are multiple schemas for references. Each schema generates its own set of references using a Python class and some optional settings given by the user (say a specific bank's account number). Each schema may only reference an invoice once. Though each schema has its own set of references, the references-space is shared between all schemas. In other words, reference 'foo' of schema A would conflict with reference 'foo' of schema B. This is because we do not know which schema was used when we encounter a reference. In reality this should not be a problem as reference schemes provided by banks usually cover a very large space, so multiple schemas are expected to just generate random numbers until one is found that has not been used yet (which should almost always happen on the first try). .. py:attribute:: __tablename__ :value: 'invoice_references' .. py:attribute:: reference :type: sqlalchemy.Column[str] .. py:attribute:: invoice_id :type: sqlalchemy.Column[uuid.UUID] .. py:attribute:: invoice :type: sqlalchemy.orm.relationship[activity.models.invoice.Invoice] .. py:attribute:: schema :type: sqlalchemy.Column[str] .. py:attribute:: bucket :type: sqlalchemy.Column[str] .. py:attribute:: __table_args__ .. py:method:: validate_schema(field: str, value: str) -> str .. py:property:: readable :type: str Returns the human formatted variant of the reference. .. py:class:: Schema(**config: object) Defines the methods that need to be implemented by schemas. Schemas should generate numbers and be able to format them. Schemas should never be deleted as we want to be able to display past schemas even if a certain schema is no longer in use. If a new schema comes along that replaces an old one in an incompatible way, the new schema should get a new name and should be added alongside the old one. .. py:attribute:: name :type: ClassVar[str] .. py:method:: __init_subclass__(name: str, **kwargs: object) -> None :classmethod: .. py:attribute:: config .. py:property:: bucket :type: str Generates a unique identifer for the current schema and config. .. py:method:: render_bucket(schema_name: str, schema_config: dict[str, Any] | None = None) -> str :classmethod: .. py:method:: link(session: sqlalchemy.orm.Session, invoice: activity.models.invoice.Invoice, optimistic: bool = False, flush: bool = True) -> InvoiceReference | None Creates a new :class:`InvoiceReference` for the given invoice. The returned invoice should have a unique reference, so the chance of ending up with a conflict error later down the line are slim. If the schema already has a linke to the invoice, we skip the creation. By default we check our constraints before we write to the database. To be faster in performance critical situation we can however also chose to be 'optimistic' and forego those checks. Due to the random nature of schema references this should usually work. The constraints are set on the database, so they will be enforced either way. Additionally we can forego the session.flush if we want to. .. py:method:: new() -> str :abstractmethod: Returns a new reference in the most compact way possible. .. py:method:: format(reference: str) -> str :abstractmethod: Turns the reference into something human-readable. .. py:class:: FeriennetSchema(**config: object) Bases: :py:obj:`Schema` The default schema for customers without specific bank integrations. The generated reference is entered as a note when conducting the online-banking transaction. .. py:method:: new() -> str Returns a new reference in the most compact way possible. .. py:method:: format(reference: str) -> str Turns the reference into something human-readable. .. py:method:: extract(text: str | None) -> str | None Takes a bunch of text and tries to extract the feriennet-v1 reference from it. .. py:class:: ESRSchema(**config: object) Bases: :py:obj:`Schema` The default schema for ESR by Postfinance. In it's default form it is random and requires no configuration. A ESR reference has 27 characters from 0-9. The first 26 characters can be chosen freely and the last character is the checksum. Some banks require that references have certain prefixes/suffixes, but Postfinance who defined the ESR standard does not. .. py:method:: new() -> str Returns a new reference in the most compact way possible. .. py:method:: checksum(number: str) -> str Generates the modulo 10 checksum as required by Postfinance. .. py:method:: format(reference: str) -> str Takes an ESR reference and formats it in a human-readable way. This is mandated as follows by Postfinance: > Die Referenznummer ist rechtsbündig, in 5er-Blocks und einem > allfälligen Restblock zu platzieren. Vorlaufende Nullen können > unterdrückt werden. .. py:class:: RaiffeisenSchema(**config: object) Bases: :py:obj:`ESRSchema` Customised ESR for Raiffeisen. .. py:attribute:: esr_identification_number :type: str .. py:method:: new() -> str Returns a new reference in the most compact way possible.