Source code for rattail.db.model.products

# -*- coding: utf-8; -*-
################################################################################
#
#  Rattail -- Retail Software Framework
#  Copyright © 2010-2023 Lance Edgar
#
#  This file is part of Rattail.
#
#  Rattail is free software: you can redistribute it and/or modify it under the
#  terms of the GNU General Public License as published by the Free Software
#  Foundation, either version 3 of the License, or (at your option) any later
#  version.
#
#  Rattail 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 General Public License for more
#  details.
#
#  You should have received a copy of the GNU General Public License along with
#  Rattail.  If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
"""
Data Models for Products
"""

import datetime
import logging

import sqlalchemy as sa
from sqlalchemy import orm
from sqlalchemy.ext.orderinglist import ordering_list
from sqlalchemy.orm.collections import attribute_mapped_collection
from sqlalchemy.ext.associationproxy import association_proxy

from rattail import enum
from rattail.db.model import (Base, uuid_column, getset_factory,
                              Department, Subdepartment, Category, Vendor, Store)
from rattail.db.types import GPCType
from rattail.db.util import make_full_description
from rattail.util import pretty_quantity


log = logging.getLogger(__name__)


[docs] class UnitOfMeasure(Base): """ Maps a UOM abbreviation used by the organization, to the proper UOM code used internally by Rattail. Note that the SIL code is similar, but apparently different, from the `codes defined by GS1`_. .. _codes defined by GS1: https://resources.gs1us.org/GS1-US-Data-Hub-Help-Center/ArtMID/3451/ArticleID/116/Unit-of-Measure-Codes """ __tablename__ = 'unit_of_measure' __table_args__ = ( sa.UniqueConstraint('abbreviation', name='unit_of_measure_uq_abbreviation'), ) __versioned__ = {} model_title = "Unit of Measure" model_title_plural = "Units of Measure" uuid = uuid_column() abbreviation = sa.Column(sa.String(length=20), nullable=False, doc=""" UOM abbreviation as it is used by the organization, e.g. 'OZ'. """) sil_code = sa.Column(sa.String(length=4), nullable=True, doc=""" SIL code for the UOM, as used internally by Rattail, e.g. '48'. """) notes = sa.Column(sa.Text(), nullable=True, doc=""" Misc. notes for this mapping. """) def __str__(self): return self.abbreviation or ""
[docs] class Brand(Base): """ Represents a brand or similar product line. """ __tablename__ = 'brand' __versioned__ = {} uuid = uuid_column() name = sa.Column(sa.String(length=100), nullable=False, doc=""" Brand name, as seen on product packaging etc. """) confirmed = sa.Column(sa.Boolean(), nullable=True, doc=""" Flag indicating the brand has been "confirmed good" by someone responsible for product maintenance (if applicable). In some cases the source of brand data is a bit "loose" and it can be helpful to have a human review and de-duplicate the brand listing. """) def __str__(self): return self.name or ''
[docs] class Tax(Base): """ Represents a sales tax rate to be applied to products. """ __tablename__ = 'tax' __table_args__ = ( sa.UniqueConstraint('code', name='tax_uq_code'), ) __versioned__ = {} uuid = uuid_column() code = sa.Column(sa.String(length=30), nullable=False, doc=""" Unique "code" for the tax rate. """) description = sa.Column(sa.String(length=255), nullable=True, doc=""" Human-friendly description for the tax rate. """) rate = sa.Column(sa.Numeric(precision=7, scale=5), nullable=False, doc=""" Percentage rate for the tax, e.g. 8.25. """) def __str__(self): if self.rate is not None: return f"{self.description} {self.rate:0.3f} %" return self.description or ''
[docs] class Product(Base): """ Represents a product for sale and/or purchase. """ __tablename__ = 'product' __table_args__ = ( sa.ForeignKeyConstraint(['unit_uuid'], ['product.uuid'], name='product_fk_unit'), sa.ForeignKeyConstraint(['department_uuid'], ['department.uuid'], name='product_fk_department'), sa.ForeignKeyConstraint(['subdepartment_uuid'], ['subdepartment.uuid'], name='product_fk_subdepartment'), sa.ForeignKeyConstraint(['category_uuid'], ['category.uuid'], name='product_fk_category'), sa.ForeignKeyConstraint(['family_uuid'], ['family.uuid'], name='product_fk_family'), sa.ForeignKeyConstraint(['brand_uuid'], ['brand.uuid'], name='product_fk_brand'), sa.ForeignKeyConstraint(['report_code_uuid'], ['report_code.uuid'], name='product_fk_report_code'), sa.ForeignKeyConstraint(['deposit_link_uuid'], ['deposit_link.uuid'], name='product_fk_deposit_link'), sa.ForeignKeyConstraint(['tax_uuid'], ['tax.uuid'], name='product_fk_tax'), sa.ForeignKeyConstraint(['suggested_price_uuid'], ['product_price.uuid'], name='product_fk_suggested_price', use_alter=True), sa.ForeignKeyConstraint(['regular_price_uuid'], ['product_price.uuid'], name='product_fk_regular_price', use_alter=True), sa.ForeignKeyConstraint(['current_price_uuid'], ['product_price.uuid'], name='product_fk_current_price', use_alter=True), sa.ForeignKeyConstraint(['tpr_price_uuid'], ['product_price.uuid'], name='product_fk_tpr_price', use_alter=True), sa.ForeignKeyConstraint(['sale_price_uuid'], ['product_price.uuid'], name='product_fk_sale_price', use_alter=True), sa.Index('product_ix_upc', 'upc'), ) __versioned__ = {'exclude': ['last_sold']} uuid = uuid_column() upc = sa.Column(GPCType(), nullable=True, doc=""" Proper GPC value for the product, if any. """) scancode = sa.Column(sa.String(length=14), nullable=True, doc=""" Scan code (as string) for the product, if any. This is generally intended to match the :attr:`upc` value, though that is not enforced. """) item_id = sa.Column(sa.String(length=50), nullable=True, doc=""" Generic ID string for the item. """) item_type = sa.Column(sa.Integer(), nullable=True, doc=""" Item type code as integer. """) department_uuid = sa.Column(sa.String(length=32)) subdepartment_uuid = sa.Column(sa.String(length=32)) category_uuid = sa.Column(sa.String(length=32)) family_uuid = sa.Column(sa.String(length=32)) report_code_uuid = sa.Column(sa.String(length=32), nullable=True, doc=""" UUID of the product's report code, if any. """) deposit_link_uuid = sa.Column(sa.String(length=32), nullable=True, doc=""" UUID of the product's deposit link, if any. """) deposit_link = orm.relationship('DepositLink', doc=""" Reference to the :class:`DepositLink` instance with which the product associates, if any. """) tax_uuid = sa.Column(sa.String(length=32), nullable=True, doc=""" UUID of the product's tax, if any. """) tax = orm.relationship(Tax, doc=""" Reference to the :class:`Tax` instance with which the product associates, if any. """) brand_uuid = sa.Column(sa.String(length=32)) description = sa.Column(sa.String(length=255), nullable=True, doc=""" Primary description for the item. """) description2 = sa.Column(sa.String(length=255), nullable=True, doc=""" Secondary description for the item. """) size = sa.Column(sa.String(length=30), nullable=True, doc=""" Free-form / human-friendly size string for the product, e.g. '32 oz' """) unit_size = sa.Column(sa.Numeric(precision=8, scale=3), nullable=True, doc=""" Unit size for product, as decimal. This refers to the numeric size of a unit of the product, in terms of the :attr:`unit_of_measure`, and may be used for smarter price comparisons etc. """) unit_of_measure = sa.Column(sa.String(length=4), nullable=False, default=enum.UNIT_OF_MEASURE_NONE, doc=""" Code indicating the unit of measure for the product. Value should be one of the keys of the ``rattail.enum.UNIT_OF_MEASURE`` dictionary. """) uom_abbreviation = sa.Column(sa.String(length=10), nullable=True, doc=""" Optional abbreviated unit of measure. This value is intended to contain a human-friendly abbreviation for the :attr:`unit_of_measure`. This value is generally also seen at the end of :attr:`size`. """) weighed = sa.Column(sa.Boolean(), nullable=False, default=False, doc=""" Whether or not the product must be weighed to determine its final price. """) average_weight = sa.Column(sa.Numeric(precision=12, scale=3), nullable=True, doc=""" Average weight of a single item "unit", e.g. the weight of one single apple. Useful for estimating final weight for when a customer orders 3 apples, etc. """) unit_uuid = sa.Column(sa.String(length=32), nullable=True) unit = orm.relationship( 'Product', remote_side=[uuid], doc=""" Reference to the *unit* product upon which this product is based, if any. """, backref=orm.backref( 'packs', cascade_backrefs=False, doc=""" List of products which reference the current product as their unit. """)) pack_size = sa.Column(sa.Numeric(precision=9, scale=4), nullable=True, doc=""" Number of *units* which constitute the current *pack* product, if applicable. """) default_pack = sa.Column(sa.Boolean(), nullable=True, doc=""" If set, this flag indicates the product is the "default pack" for its unit item. This flag is only relevant if the product is in fact a pack item; it should be effectively ignored for a unit item. """) case_size = sa.Column(sa.Numeric(precision=9, scale=4), nullable=True, doc=""" Case size for the product, i.e. how many units (current product) are included when *sold* as a case. """) organic = sa.Column(sa.Boolean(), nullable=False, default=False, doc=""" Whether the item is organic. """) kosher = sa.Column(sa.Boolean(), nullable=True, doc=""" Whether the item is kosher. """) vegan = sa.Column(sa.Boolean(), nullable=True, doc=""" Whether the item is vegan. """) vegetarian = sa.Column(sa.Boolean(), nullable=True, doc=""" Whether the item is vegetarian. """) gluten_free = sa.Column(sa.Boolean(), nullable=True, doc=""" Whether the item is gluten-free. """) sugar_free = sa.Column(sa.Boolean(), nullable=True, doc=""" Whether the item is sugar-free. """) not_for_sale = sa.Column(sa.Boolean(), nullable=False, default=False, doc=""" Flag to indicate items which are not available for sale. """) discontinued = sa.Column(sa.Boolean(), nullable=False, default=False, doc=""" Flag to indicate an item has been discontinued. """) deleted = sa.Column(sa.Boolean(), nullable=False, default=False, doc=""" Flag to indicate items which have been deleted. Obviously this is implies "false" deletion, where the record is actually kept on file. Whether or not you use this is up to you. """) price_required = sa.Column(sa.Boolean(), nullable=True, doc=""" Indicates that the price must be manually entered by cashier, at check-out. """) suggested_price_uuid = sa.Column(sa.String(length=32), nullable=True) regular_price_uuid = sa.Column(sa.String(length=32), nullable=True) current_price_uuid = sa.Column(sa.String(length=32), nullable=True) tpr_price_uuid = sa.Column(sa.String(length=32), nullable=True) tpr_price = orm.relationship( 'ProductPrice', primaryjoin='Product.tpr_price_uuid == ProductPrice.uuid', lazy='joined', post_update=True) sale_price_uuid = sa.Column(sa.String(length=32), nullable=True) sale_price = orm.relationship( 'ProductPrice', primaryjoin='Product.sale_price_uuid == ProductPrice.uuid', lazy='joined', post_update=True) discountable = sa.Column(sa.Boolean(), nullable=False, default=True, doc=""" Whether or not the product may be discounted in any way. """) special_order = sa.Column(sa.Boolean(), nullable=False, default=False, doc=""" Whether the product is available for special order. """) food_stampable = sa.Column(sa.Boolean(), nullable=True, doc=""" Flag indicating whether food stamps are a valid form of payment for the item. """) tax1 = sa.Column(sa.Boolean(), nullable=True, doc=""" (hack?) Flag indicating whether 'tax 1' applies to the item. """) tax2 = sa.Column(sa.Boolean(), nullable=True, doc=""" (hack?) Flag indicating whether 'tax 2' applies to the item. """) tax3 = sa.Column(sa.Boolean(), nullable=True, doc=""" (hack?) Flag indicating whether 'tax 3' applies to the item. """) last_sold = sa.Column(sa.DateTime(), nullable=True, doc=""" UTC timestamp of the product's last sale event. """) ingredients = sa.Column(sa.Text(), nullable=True, doc=""" Free-form ingredients for the product. """) status_code = sa.Column(sa.Integer(), nullable=True, doc=""" Status code for the product. Interpretation of this value is left to the custom app logic. """) notes = sa.Column(sa.Text(), nullable=True, doc=""" Generic / free-form notes for the product. """) department = orm.relationship(Department, order_by=Department.name) subdepartment = orm.relationship( Subdepartment, order_by=Subdepartment.name, backref=orm.backref( 'products', cascade_backrefs=False)) category = orm.relationship( Category, backref=orm.backref( 'products', cascade_backrefs=False)) brand = orm.relationship( Brand, doc=""" Reference to the brand object, if any. """, backref=orm.backref( 'products', cascade_backrefs=False, doc=""" List of products for the brand. """)) family = orm.relationship( 'Family', doc=""" Reference to the :class:`Family` instance with which the product associates, if any. """, backref=orm.backref( 'products', cascade_backrefs=False, doc=""" List of :class:`Product` objects which belong to this family. """)) report_code = orm.relationship( 'ReportCode', doc=""" Reference to the :class:`ReportCode` instance with which the product associates, if any. """, backref=orm.backref( 'products', cascade_backrefs=False, doc=""" List of :class:`Product` objects which associate with this report code. """)) def __str__(self): return self.full_description @property def full_description(self): """ Convenience attribute which returns a more complete description. Most notably, this includes the brand name and product size. """ return make_full_description(self.brand.name if self.brand else '', self.description, self.size)
[docs] def is_unit_item(self): """ Returns True if the product is considered a "unit" item, as opposed to a "pack" item. """ if self.unit: # we appear to be a pack for some other unit item return False return True
[docs] def is_pack_item(self): """ Returns True if the product is considered a "pack" item, as opposed to a "unit" item. """ if self.unit: return True return False
[docs] def get_unit_item(self): """ Returns the *unit* item for the current product. If this product *is* a unit item, will return ``self``; otherwise returns ``self.unit``. """ if self.is_unit_item(): return self return self.unit
[docs] def get_default_pack_item(self): """ Returns the "default" pack item for the current product, which is assumed to be a unit item. """ if self.is_unit_item(): for pack in self.packs: if pack.default_pack: return pack if self.packs: log.warning("unit item %s has %s packs, but none is default: %s", self.uuid, len(self.packs), self) return self.packs[0]
@property def pretty_upc(self): """ Product's UPC as a somewhat human-readable string. """ if self.upc is None: return None return self.upc.pretty()
[docs] def costs_for_vendor(self, vendor): """ Return all costs associated with the given vendor. """ return [cost for cost in self.costs if cost.vendor is vendor]
[docs] def cost_for_vendor(self, vendor, error_if_multiple=False, error_if_none=False): """ Locate and return the first cost associated with the given vendor. :param error_if_multiple: If set, and the product has more than one cost associated with the given vendor, an exception will be raised. :param error_if_none: If set, and the product has no cost(s) associated with the given vendor, an exception will be raised. """ costs = self.costs_for_vendor(vendor) if costs: if error_if_multiple and len(costs) > 1: raise ValueError("Product {} {} has {} costs associated with vendor: {}".format( self.upc, self.full_description, len(costs), vendor)) return costs[0] elif error_if_none: raise ValueError("Product {} {} has 0 costs associated with vendor: {}".format( self.upc, self.full_description, vendor))
[docs] class ProductImage(Base): """ Contains an image for a product. """ __tablename__ = 'product_image' __table_args__ = ( sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='product_image_fk_product'), ) uuid = uuid_column() product_uuid = sa.Column(sa.String(length=32), nullable=False) product = orm.relationship( Product, doc=""" Reference to the product which is shown by the image. """, backref=orm.backref( 'image', uselist=False, cascade_backrefs=False, doc=""" Reference to the product's image, if any. """)) bytes = sa.Column(sa.LargeBinary(), nullable=False)
[docs] class ProductCode(Base): """ Represents an arbitrary "code" for a product. """ __tablename__ = 'product_code' __table_args__ = ( sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='product_code_fk_product'), ) __versioned__ = {} uuid = uuid_column() product_uuid = sa.Column(sa.String(length=32), nullable=False) ordinal = sa.Column(sa.Integer(), nullable=False) code = sa.Column(sa.String(length=20), nullable=True, doc=""" Alternate code value for the product, to be used for lookup. """) def __str__(self): return self.code or ''
Product._codes = orm.relationship( ProductCode, backref='product', collection_class=ordering_list('ordinal', count_from=1), order_by=ProductCode.ordinal, cascade='save-update, merge, delete, delete-orphan') Product.codes = association_proxy( '_codes', 'code', getset_factory=getset_factory, creator=lambda c: ProductCode(code=c)) Product._code = orm.relationship( ProductCode, primaryjoin=sa.and_( ProductCode.product_uuid == Product.uuid, ProductCode.ordinal == 1, ), uselist=False, viewonly=True) Product.code = association_proxy( '_code', 'code', getset_factory=getset_factory)
[docs] class ProductCost(Base): """ Represents a source from which a product may be obtained via purchase. """ __tablename__ = 'product_cost' __table_args__ = ( sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='product_cost_fk_product'), sa.ForeignKeyConstraint(['vendor_uuid'], ['vendor.uuid'], name='product_cost_fk_vendor'), ) __versioned__ = {} uuid = uuid_column() product_uuid = sa.Column(sa.String(length=32), nullable=False) vendor_uuid = sa.Column(sa.String(length=32), nullable=False) preference = sa.Column(sa.Integer(), nullable=False) code = sa.Column(sa.String(length=20), nullable=True, doc=""" Vendor order code for the item, if applicable. """) case_size = sa.Column(sa.Numeric(precision=9, scale=4), nullable=True, doc=""" Number of units which constitute a "case" in the context of this vendor catalog item. May be a fractional quantity, e.g. 17.5LB. """) # TODO: we're using precision=10, scale=5 elsewhere; should grow this too case_cost = sa.Column(sa.Numeric(precision=9, scale=5), nullable=True, doc=""" Case cost for the item, if known. """) # TODO: we're using precision=10, scale=5 elsewhere; should grow this too unit_cost = sa.Column(sa.Numeric(precision=9, scale=5), nullable=True, doc=""" Unit cost for the item, if known. """) pack_size = sa.Column(sa.Integer()) pack_cost = sa.Column(sa.Numeric(precision=9, scale=5)) effective = sa.Column(sa.DateTime(), nullable=True, doc=""" Date and time when the cost became effective, if known. """) discount_starts = sa.Column(sa.DateTime(), nullable=True, doc=""" Date and time when the "current" discount for the cost begins, if applicable. Note that this should never be a "future" value, since only the current discount should be part of this record. """) discount_ends = sa.Column(sa.DateTime(), nullable=True, doc=""" Date and time when the "current" discount for the cost ends, if applicable. Note that this should never be a "past" value, since only the current discount should be part of this record. """) discount_amount = sa.Column(sa.Numeric(precision=9, scale=5), nullable=True, doc=""" Dollar amount of the "current" discount for the cost, if applicable. """) discount_percent = sa.Column(sa.Numeric(precision=7, scale=4), nullable=True, doc=""" Percentage amount of the "current" discount for the cost, if applicable. """) discount_cost = sa.Column(sa.Numeric(precision=9, scale=5), nullable=True, doc=""" Discounted (effective) unit cost for the item, while current discount is in effect (if applicable). """) discontinued = sa.Column(sa.Boolean(), nullable=True, doc=""" Flag to indicate if the cost record has been discontinued, presumably by the vendor. """) vendor = orm.relationship( Vendor, backref=orm.backref( 'product_costs', cascade='all', cascade_backrefs=False))
Product.costs = orm.relationship( ProductCost, backref='product', collection_class=ordering_list('preference', count_from=1), order_by=ProductCost.preference, cascade='save-update, merge, delete, delete-orphan') Product.cost = orm.relationship( ProductCost, primaryjoin=sa.and_( ProductCost.product_uuid == Product.uuid, ProductCost.preference == 1, ), uselist=False, viewonly=True) Product.vendor = orm.relationship( Vendor, secondary=ProductCost.__table__, primaryjoin=sa.and_( ProductCost.product_uuid == Product.uuid, ProductCost.preference == 1, ), secondaryjoin=Vendor.uuid == ProductCost.vendor_uuid, uselist=False, viewonly=True)
[docs] class ProductFutureCost(Base): """ Represents a *future* cost record, i.e. an otherwise normal / complete cost record, but which will become effective at some later date/time. Note that this is meant to be more of a "queue" table in practice, i.e. records which exist here should eventually be applied to the :class:`ProductCost` table, at which point the future record(s) would be deleted. """ __tablename__ = 'product_future_cost' __table_args__ = ( sa.ForeignKeyConstraint(['cost_uuid'], ['product_cost.uuid'], name='product_future_cost_fk_cost'), sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='product_future_cost_fk_product'), sa.ForeignKeyConstraint(['vendor_uuid'], ['vendor.uuid'], name='product_future_cost_fk_vendor'), ) uuid = uuid_column() cost_uuid = sa.Column(sa.String(length=32), nullable=True) cost = orm.relationship( ProductCost, doc=""" Reference to the "current" cost record to which this future cost record will be applied (when the time comes), if applicable. If this is ``None``, then a new "current" cost record would be created instead. """, backref=orm.backref( 'futures', cascade_backrefs=False)) product_uuid = sa.Column(sa.String(length=32), nullable=False) product = orm.relationship( Product, doc=""" Reference to the product to which the cost record pertains. """, backref=orm.backref( 'future_costs', cascade_backrefs=False, doc=""" Sequence of future cost records for the product, i.e. which have yet to become "current" costs. """)) vendor_uuid = sa.Column(sa.String(length=32), nullable=False) vendor = orm.relationship( Vendor, doc=""" Reference to the vendor to which the cost record pertains. """, backref=orm.backref( 'future_costs', cascade_backrefs=False, doc=""" Sequence of future cost records for the vendor, i.e. which have yet to become "current" costs. """)) # TODO: we don't want this to be part of it, right? (at least not yet) # preference = sa.Column(sa.Integer(), nullable=False) order_code = sa.Column(sa.String(length=20), nullable=True, doc=""" Vendor order code for the item, if applicable. """) case_quantity = sa.Column(sa.Numeric(precision=9, scale=4), nullable=True, doc=""" Number of units which constitute a "case" in the context of this vendor catalog item. May be a fractional quantity, e.g. 17.5LB. """) case_cost = sa.Column(sa.Numeric(precision=9, scale=5), nullable=True, doc=""" Case cost for the item, if known. """) unit_cost = sa.Column(sa.Numeric(precision=9, scale=5), nullable=True, doc=""" Unit cost for the item, if known. """) starts = sa.Column(sa.DateTime(), nullable=False, doc=""" Date and time when the cost becomes effective. """) ends = sa.Column(sa.DateTime(), nullable=True, doc=""" Date and time when the cost *stops* being effective, if applicable. This often will be null, in which case the cost becomes "permanently" effective, i.e. until a newer cost is brought in. """) discontinued = sa.Column(sa.Boolean(), nullable=True, doc=""" Flag to indicate if the cost record has been discontinued, presumably by the vendor. """)
[docs] class ProductPrice(Base): """ Represents a price for a product. """ __tablename__ = 'product_price' __table_args__ = ( sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='product_price_fk_product'), ) __versioned__ = {} uuid = uuid_column() product_uuid = sa.Column(sa.String(length=32), nullable=False) type = sa.Column(sa.Integer()) level = sa.Column(sa.Integer()) starts = sa.Column(sa.DateTime()) ends = sa.Column(sa.DateTime()) price = sa.Column(sa.Numeric(precision=8, scale=3)) multiple = sa.Column(sa.Integer()) pack_price = sa.Column(sa.Numeric(precision=8, scale=3)) pack_multiple = sa.Column(sa.Integer()) def __str__(self): text = "" if self.price: text = f"${self.price:0.2f}" if self.multiple: text += f"/{self.multiple}" if self.pack_price: pack_text = f"${self.pack_price:0.2f}" if self.pack_multiple: pack_text += f"/{self.pack_multiple}" text += f" ({pack_text})" return text
[docs] def active_now(self): """ Returns boolean indicating whether the price is currently active, i.e. "now" falls within time window defined by :attr:`starts` and :attr:`ends`. Note that if the price has no start and/or end time, this will return ``True``. """ now = datetime.datetime.utcnow() if self.starts and self.ends: return self.starts <= now <= self.ends elif self.starts: return self.starts <= now elif self.ends: return now <= self.ends return True
Product.prices = orm.relationship( ProductPrice, backref='product', primaryjoin=ProductPrice.product_uuid == Product.uuid, cascade='save-update, merge, delete, delete-orphan') Product.suggested_price = orm.relationship( ProductPrice, primaryjoin=Product.suggested_price_uuid == ProductPrice.uuid, lazy='joined', post_update=True) Product.regular_price = orm.relationship( ProductPrice, primaryjoin=Product.regular_price_uuid == ProductPrice.uuid, lazy='joined', post_update=True) Product.current_price = orm.relationship( ProductPrice, primaryjoin=Product.current_price_uuid == ProductPrice.uuid, lazy='joined', post_update=True)
[docs] class ProductInventory(Base): """ Inventory data for a product. Assumption at this point is that this data will be accurate only in the context of the "local" node (store etc.). Tracking inventory for multiple nodes is not yet supported in the "host" sense; however each node may track its own inventory (only). """ __tablename__ = 'product_inventory' __table_args__ = ( sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='product_inventory_fk_product'), ) uuid = uuid_column() product_uuid = sa.Column(sa.String(length=32), nullable=False) product = orm.relationship( Product, doc=""" Product to which this inventory record pertains. """, backref=orm.backref( 'inventory', uselist=False, cascade='all, delete-orphan', cascade_backrefs=False, doc=""" Inventory data for the product, if any. """)) on_hand = sa.Column(sa.Numeric(precision=9, scale=4), nullable=True, doc=""" Unit quantity of product which is currently on hand. """) on_order = sa.Column(sa.Numeric(precision=9, scale=4), nullable=True, doc=""" Unit quantity of product which is currently on order. """)
[docs] class ProductStoreInfo(Base): """ General store-specific info for a product. """ __tablename__ = 'product_store_info' __table_args__ = ( sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='product_store_info_fk_product'), sa.ForeignKeyConstraint(['store_uuid'], ['store.uuid'], name='product_store_info_fk_store'), ) uuid = uuid_column() product_uuid = sa.Column(sa.String(length=32), nullable=False) product = orm.relationship( Product, doc=""" Product to which this info record pertains. """, backref=orm.backref( 'store_infos', collection_class=attribute_mapped_collection('store_uuid'), cascade='all, delete-orphan', cascade_backrefs=False, doc=""" List of store-specific info records for the product. """)) store_uuid = sa.Column(sa.String(length=32), nullable=False) store = orm.relationship( Store, doc=""" Store to which this info record pertains. """) recently_active = sa.Column(sa.Boolean(), nullable=True, doc=""" Flag indicating the product has seen "recent activity" at the store. How this is populated and/or interpreted is up to custom app logic. """)
[docs] class ProductVolatile(Base): """ This is the place to find "volatile" data for a given product, or at least it should be... As of this writing there are a couple other places to look but hopefully this table can eventually be "the" place. Whether any given value in a given record, applies to the "current" app node only, or if it applies to all nodes, is up to app logic. Note that a custom app should (most likely) *not* bother "extending" this table, but rather should create a separate table with similar pattern. """ __tablename__ = 'product_volatile' __table_args__ = ( sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='product_volatile_fk_product'), ) uuid = uuid_column() product_uuid = sa.Column(sa.String(length=32), nullable=False) product = orm.relationship( Product, doc=""" Product to which this "volatile" data record pertains. """, backref=orm.backref( 'volatile', uselist=False, cascade='all, delete-orphan', cascade_backrefs=False, doc=""" "Volatile" data record for the product, if any. """)) true_cost = sa.Column(sa.Numeric(precision=9, scale=5), nullable=True, doc=""" "True" unit cost for the item, if known. This might include certain "allowances" (discounts) currently in effect etc.; really anything which might not be reflected in "official" unit cost for the product. Usually, this value is quite easily calculated and so this field serves as more of a cache, for sake of SQL access to the values. """) true_margin = sa.Column(sa.Numeric(precision=12, scale=5), nullable=True, doc=""" "True" profit margin for the "regular" unit price vs. the "true" unit cost (:attr:`true_cost`). This is expressed as a human-friendly percentage value, e.g. ``37.5`` (instead of ``0.375``) for 37.5% margin. """)
[docs] class PendingProduct(Base): """ A "pending" product record, used for new product entry workflow. """ __tablename__ = 'pending_product' __table_args__ = ( sa.ForeignKeyConstraint(['product_uuid'], ['product.uuid'], name='pending_product_fk_product'), sa.ForeignKeyConstraint(['department_uuid'], ['department.uuid'], name='pending_product_fk_department'), sa.ForeignKeyConstraint(['brand_uuid'], ['brand.uuid'], name='pending_product_fk_brand'), sa.ForeignKeyConstraint(['vendor_uuid'], ['vendor.uuid'], name='pending_product_fk_vendor'), sa.ForeignKeyConstraint(['user_uuid'], ['user.uuid'], name='pending_product_fk_user'), sa.ForeignKeyConstraint(['resolved_by_uuid'], ['user.uuid'], name='pending_product_fk_resolved_by'), ) uuid = uuid_column() user_uuid = sa.Column(sa.String(length=32), nullable=False) user = orm.relationship( 'User', foreign_keys=[user_uuid], doc=""" Reference to the :class:`~rattail:rattail.db.model.User` who first entered the record. """) created = sa.Column(sa.DateTime(), nullable=False, default=datetime.datetime.utcnow, doc=""" Timestamp when the record was first created. """) product_uuid = sa.Column(sa.String(length=32), nullable=True) product = orm.relationship( Product, cascade_backrefs=False, doc=""" Reference to the final :class:`~rattail.db.model.Product` record, if applicable. """) # Product fields upc = sa.Column(GPCType(), nullable=True) scancode = sa.Column(sa.String(length=14), nullable=True) item_id = sa.Column(sa.String(length=50), nullable=True) item_type = sa.Column(sa.Integer(), nullable=True) department_name = sa.Column(sa.String(length=30), nullable=True) department_uuid = sa.Column(sa.String(length=32), nullable=True) department = orm.relationship(Department) brand_name = sa.Column(sa.String(length=100), nullable=True) brand_uuid = sa.Column(sa.String(length=32), nullable=True) brand = orm.relationship(Brand) description = sa.Column(sa.String(length=255), nullable=True) size = sa.Column(sa.String(length=30), nullable=True) vendor_name = sa.Column(sa.String(length=50), nullable=True) vendor_uuid = sa.Column(sa.String(length=32), nullable=True) vendor = orm.relationship(Vendor) vendor_item_code = sa.Column(sa.String(length=20), nullable=True) unit_cost = sa.Column(sa.Numeric(precision=10, scale=5), nullable=True) case_size = sa.Column(sa.Numeric(precision=9, scale=4), nullable=True) regular_price_amount = sa.Column(sa.Numeric(precision=8, scale=3), nullable=True) special_order = sa.Column(sa.Boolean(), nullable=True) notes = sa.Column(sa.Text(), nullable=True) # workflow fields status_code = sa.Column(sa.Integer(), nullable=True, doc=""" Status indicator for the new product record. """) resolved = sa.Column(sa.DateTime(), nullable=True, doc=""" Timestamp when the record was resolved. """) resolved_by_uuid = sa.Column(sa.String(length=32), nullable=True) resolved_by = orm.relationship( 'User', foreign_keys=[resolved_by_uuid], doc=""" Reference to the :class:`~rattail:rattail.db.model.User` who resolved the record. """) @property def full_description(self): return make_full_description(self.brand_name, self.description, self.size) def __str__(self): return self.full_description
[docs] class InventoryAdjustmentReason(Base): """ Reasons for adjusting product inventory. """ __tablename__ = 'invadjust_reason' __table_args__ = ( sa.UniqueConstraint('code', name='invadjust_reason_uq_code'), ) uuid = uuid_column() code = sa.Column(sa.String(length=20), nullable=False, doc=""" Unique code for the reason. """) description = sa.Column(sa.String(length=255), nullable=False, doc=""" Description for the reason. """) hidden = sa.Column(sa.Boolean(), nullable=True, doc=""" Flag indicating that the reason code should *not* be generally visible for selection by the user etc. """) def __str__(self): return self.description or ""