tailbone/tailbone/views/batch/vendorcatalog.py
2023-02-21 17:35:47 -06:00

453 lines
15 KiB
Python

# -*- 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/>.
#
################################################################################
"""
Views for maintaining vendor catalogs
"""
import logging
from rattail.db import model
import colander
from deform import widget as dfwidget
from webhelpers2.html import tags
from tailbone import forms
from tailbone.views.batch import FileBatchMasterView
from tailbone.diffs import Diff
from tailbone.db import Session
log = logging.getLogger(__name__)
class VendorCatalogView(FileBatchMasterView):
"""
Master view for vendor catalog batches.
"""
model_class = model.VendorCatalogBatch
model_row_class = model.VendorCatalogBatchRow
default_handler_spec = 'rattail.batch.vendorcatalog:VendorCatalogHandler'
route_prefix = 'vendorcatalogs'
url_prefix = '/vendors/catalogs'
template_prefix = '/batch/vendorcatalog'
bulk_deletable = True
results_executable = True
rows_bulk_deletable = True
has_input_file_templates = True
configurable = True
labels = {
'vendor_id': "Vendor ID",
'parser_key': "Parser",
}
grid_columns = [
'id',
'vendor',
'description',
'filename',
'rowcount',
'created',
'executed',
]
form_fields = [
'id',
'filename',
'parser_key',
'vendor',
'future',
'effective',
'cache_products',
'params',
'description',
'notes',
'created',
'created_by',
'rowcount',
'executed',
'executed_by',
]
row_grid_columns = [
'sequence',
'upc',
'brand_name',
'description',
'size',
'vendor_code',
'is_preferred_vendor',
'old_unit_cost',
'unit_cost',
'unit_cost_diff',
'unit_cost_diff_percent',
'starts',
'status_code',
]
row_form_fields = [
'sequence',
'item_entry',
'product',
'upc',
'brand_name',
'description',
'size',
'is_preferred_vendor',
'suggested_retail',
'starts',
'ends',
'discount_starts',
'discount_ends',
'discount_amount',
'discount_percent',
'case_cost_diff',
'unit_cost_diff',
'status_code',
'status_text',
]
def get_input_file_templates(self):
return [
{'key': 'default',
'label': "Default",
'default_url': self.request.static_url(
'tailbone:static/files/vendor_catalog_template.xlsx')},
]
def get_parsers(self):
if not hasattr(self, 'parsers'):
app = self.get_rattail_app()
vendor_handler = app.get_vendor_handler()
self.parsers = vendor_handler.get_supported_catalog_parsers()
return self.parsers
def configure_grid(self, g):
super(VendorCatalogView, self).configure_grid(g)
model = self.model
# nb. this batch has vendor_id and vendor_name fields, but in
# practice they aren't used much and normally just the vendor
# proper is set. so we remove simple filters and add the
# custom one for (referenced) vendor name
g.remove_filter('vendor_id')
g.remove_filter('vendor_name')
g.set_joiner('vendor', lambda q: q.join(model.Vendor))
g.set_filter('vendor', model.Vendor.name,
default_active=True,
default_verb='contains')
# and this is the same thing we are showing as grid column
g.set_sorter('vendor', model.Vendor.name)
# preferred filters
g.set_filters_sequence([
'id',
'vendor',
'description',
'executed',
'created',
'filename',
'future',
'effective',
'notes',
])
g.set_link('vendor')
g.set_link('filename')
def configure_form(self, f):
super(VendorCatalogView, self).configure_form(f)
app = self.get_rattail_app()
vendor_handler = app.get_vendor_handler()
# filename
f.set_label('filename', "Catalog File")
# parser_key
if self.creating:
if 'parser_key' not in f:
f.insert_after('filename', 'parser_key')
parsers = self.get_parsers()
values = [(p.key, p.display) for p in parsers]
if len(values) == 1:
f.set_default('parser_key', parsers[0].key)
f.set_widget('parser_key', dfwidget.SelectWidget(values=values))
else:
f.set_readonly('parser_key')
f.set_renderer('parser_key', self.render_parser_key)
# vendor
f.set_renderer('vendor', self.render_vendor)
if self.creating and 'vendor' in f:
f.replace('vendor', 'vendor_uuid')
f.set_label('vendor_uuid', "Vendor")
f.set_required('vendor_uuid')
f.set_validator('vendor_uuid', self.valid_vendor_uuid)
# should we use dropdown or autocomplete? note that if
# autocomplete is to be used, we also must make sure we
# have an autocomplete url registered
use_dropdown = vendor_handler.choice_uses_dropdown()
if not use_dropdown:
try:
vendors_url = self.request.route_url('vendors.autocomplete')
except KeyError:
use_dropdown = True
if use_dropdown:
vendors = self.Session.query(model.Vendor)\
.order_by(model.Vendor.id)
vendor_values = [(vendor.uuid, "({}) {}".format(vendor.id,
vendor.name))
for vendor in vendors]
f.set_widget('vendor_uuid',
dfwidget.SelectWidget(values=vendor_values))
else:
vendor_display = ""
if self.request.method == 'POST':
if self.request.POST.get('vendor_uuid'):
vendor = self.Session.get(model.Vendor,
self.request.POST['vendor_uuid'])
if vendor:
vendor_display = str(vendor)
f.set_widget('vendor_uuid',
forms.widgets.JQueryAutocompleteWidget(
field_display=vendor_display,
service_url=vendors_url,
ref='vendorAutocomplete',
assigned_label='vendorName',
input_callback='vendorChanged',
new_label_callback='vendorLabelChanging'))
else:
f.set_readonly('vendor')
if self.batch_handler.allow_future():
# effective
f.set_type('effective', 'date_jquery')
else: # future not allowed
f.remove('future',
'effective')
if self.creating:
f.set_node('cache_products', colander.Boolean())
f.set_type('cache_products', 'boolean')
f.set_helptext('cache_products',
"If set, will pre-cache all products for quicker "
"lookups when loading the catalog.")
else:
f.remove('cache_products')
def valid_vendor_uuid(self, node, value):
model = self.model
if value:
vendor = self.Session.get(model.Vendor, value)
if not vendor:
raise colander.Invalid(node, "Vendor not found")
def render_parser_key(self, batch, field):
key = getattr(batch, field)
if not key:
return
app = self.get_rattail_app()
vendor_handler = app.get_vendor_handler()
parser = vendor_handler.get_catalog_parser(key)
return parser.display
def template_kwargs_create(self, **kwargs):
app = self.get_rattail_app()
vendor_handler = app.get_vendor_handler()
parsers = self.get_parsers()
parsers_data = {}
for parser in parsers:
pdata = {'key': parser.key,
'vendor_key': parser.vendor_key}
if parser.vendor_key:
vendor = vendor_handler.get_vendor(self.Session(),
parser.vendor_key)
if vendor:
pdata['vendor_uuid'] = vendor.uuid
pdata['vendor_name'] = vendor.name
parsers_data[parser.key] = pdata
kwargs['parsers'] = parsers
kwargs['parsers_data'] = parsers_data
return kwargs
def get_batch_kwargs(self, batch):
kwargs = super(VendorCatalogView, self).get_batch_kwargs(batch)
kwargs['parser_key'] = batch.parser_key
if batch.vendor:
kwargs['vendor'] = batch.vendor
elif batch.vendor_uuid:
kwargs['vendor_uuid'] = batch.vendor_uuid
if batch.vendor_id:
kwargs['vendor_id'] = batch.vendor_id
if batch.vendor_name:
kwargs['vendor_name'] = batch.vendor_name
if self.batch_handler.allow_future():
kwargs['future'] = batch.future
kwargs['effective'] = batch.effective
return kwargs
def save_create_form(self, form):
batch = super(VendorCatalogView, self).save_create_form(form)
batch.set_param('cache_products', form.validated['cache_products'])
return batch
def configure_row_grid(self, g):
super(VendorCatalogView, self).configure_row_grid(g)
batch = self.get_instance()
# starts
if not batch.future:
g.remove('starts')
g.set_type('old_unit_cost', 'currency')
g.set_type('unit_cost', 'currency')
g.set_type('unit_cost_diff', 'currency')
g.set_type('unit_cost_diff_percent', 'percent')
g.set_label('unit_cost_diff_percent', "Diff. %")
g.set_label('is_preferred_vendor', "Pref. Vendor")
g.set_label('upc', "UPC")
g.set_label('brand_name', "Brand")
g.set_label('old_unit_cost', "Old Cost")
g.set_label('unit_cost', "New Cost")
g.set_label('unit_cost_diff', "Diff. $")
g.set_link('upc')
g.set_link('brand')
g.set_link('description')
g.set_link('vendor_code')
def row_grid_extra_class(self, row, i):
if row.status_code == row.STATUS_PRODUCT_NOT_FOUND:
return 'warning'
if row.status_code in (row.STATUS_NEW_COST,
row.STATUS_UPDATE_COST, # TODO: deprecate/remove this one
row.STATUS_CHANGE_VENDOR_ITEM_CODE,
row.STATUS_CHANGE_CASE_SIZE,
row.STATUS_CHANGE_COST,
row.STATUS_CHANGE_PRODUCT):
return 'notice'
def configure_row_form(self, f):
super(VendorCatalogView, self).configure_row_form(f)
f.set_renderer('product', self.render_product)
f.set_type('upc', 'gpc')
f.set_type('discount_percent', 'percent')
f.set_type('suggested_retail', 'currency')
def template_kwargs_view_row(self, **kwargs):
row = kwargs['instance']
batch = row.batch
fields = [
'vendor_code',
'case_size',
'case_cost',
'unit_cost',
]
old_data = dict([(field, getattr(row, 'old_{}'.format(field)))
for field in fields])
new_data = dict([(field, getattr(row, field))
for field in fields])
kwargs['catalog_entry_diff'] = Diff(old_data, new_data, fields=fields,
monospace=True)
return kwargs
def configure_get_simple_settings(self):
settings = super(VendorCatalogView, self).configure_get_simple_settings() or []
settings.extend([
# key field
{'section': 'rattail.batch',
'option': 'vendor_catalog.allow_future',
'type': bool},
])
return settings
def configure_get_context(self):
context = super(VendorCatalogView, self).configure_get_context()
app = self.get_rattail_app()
vendor_handler = app.get_vendor_handler()
Parsers = vendor_handler.get_all_catalog_parsers()
Supported = vendor_handler.get_supported_catalog_parsers()
context['catalog_parsers'] = Parsers
context['catalog_parsers_data'] = dict([(Parser.key, Parser in Supported)
for Parser in Parsers])
return context
def configure_gather_settings(self, data):
settings = super(VendorCatalogView, self).configure_gather_settings(data)
app = self.get_rattail_app()
vendor_handler = app.get_vendor_handler()
supported = []
for Parser in vendor_handler.get_all_catalog_parsers():
name = 'catalog_parser_{}'.format(Parser.key)
if data.get(name) == 'true':
supported.append(Parser.key)
settings.append({'name': 'rattail.vendors.supported_catalog_parsers',
'value': ', '.join(supported)})
return settings
def configure_remove_settings(self):
super(VendorCatalogView, self).configure_remove_settings()
app = self.get_rattail_app()
names = [
'rattail.vendors.supported_catalog_parsers',
'tailbone.batch.vendorcatalog.supported_parsers', # deprecated
]
# nb. using thread-local session here; we do not use
# self.Session b/c it may not point to Rattail
session = Session()
for name in names:
app.delete_setting(session, name)
# TODO: deprecate / remove this
VendorCatalogsView = VendorCatalogView
def defaults(config, **kwargs):
base = globals()
VendorCatalogView = kwargs.get('VendorCatalogView', base['VendorCatalogView'])
VendorCatalogView.defaults(config)
def includeme(config):
defaults(config)