3
0
Fork 0

Compare commits

..

No commits in common. "84ab93108158c8ea4cd2b0c7269cbc229b38fd29" and "c800ebf4e4f7427cb13fc91f226b73d0eb05f72a" have entirely different histories.

11 changed files with 28 additions and 181 deletions

View file

@ -42,10 +42,9 @@ dependencies = [
"pyramid_fanstatic", "pyramid_fanstatic",
"pyramid_mako", "pyramid_mako",
"pyramid_tm", "pyramid_tm",
"SQLAlchemy-Utils",
"waitress", "waitress",
"WebHelpers2", "WebHelpers2",
"WuttJamaican[db]>=0.19.1", "WuttJamaican[db]>=0.19.0",
"zope.sqlalchemy>=1.5", "zope.sqlalchemy>=1.5",
] ]

View file

@ -155,28 +155,6 @@ class WuttaEnum(colander.Enum):
return widgets.SelectWidget(**kwargs) return widgets.SelectWidget(**kwargs)
class WuttaMoney(colander.Money):
"""
Custom schema type for "money" fields.
This is a subclass of :class:`colander:colander.Money`, but uses
the custom :class:`~wuttaweb.forms.widgets.WuttaMoneyInputWidget`
by default.
:param request: Current :term:`request` object.
"""
def __init__(self, request, *args, **kwargs):
super().__init__(*args, **kwargs)
self.request = request
self.config = self.request.wutta_config
self.app = self.config.get_app()
def widget_maker(self, **kwargs):
""" """
return widgets.WuttaMoneyInputWidget(self.request, **kwargs)
class WuttaSet(colander.Set): class WuttaSet(colander.Set):
""" """
Custom schema type for :class:`python:set` fields. Custom schema type for :class:`python:set` fields.

View file

@ -41,7 +41,6 @@ in the namespace:
""" """
import datetime import datetime
import decimal
import os import os
import colander import colander
@ -195,42 +194,6 @@ class WuttaDateTimeWidget(DateTimeInputWidget):
return super().serialize(field, cstruct, **kw) return super().serialize(field, cstruct, **kw)
class WuttaMoneyInputWidget(MoneyInputWidget):
"""
Custom widget for "money" fields. This is used by default for
:class:`~wuttaweb.forms.schema.WuttaMoney` type nodes.
The main purpose of this widget is to leverage
:meth:`~wuttjamaican:wuttjamaican.app.AppHandler.render_currency()`
for the readonly display.
This is a subclass of
:class:`deform:deform.widget.MoneyInputWidget` and uses these
Deform templates:
* ``moneyinput``
:param request: Current :term:`request` object.
"""
def __init__(self, request, *args, **kwargs):
super().__init__(*args, **kwargs)
self.request = request
self.config = self.request.wutta_config
self.app = self.config.get_app()
def serialize(self, field, cstruct, **kw):
""" """
readonly = kw.get('readonly', self.readonly)
if readonly:
if cstruct in (colander.null, None):
return ""
cstruct = decimal.Decimal(cstruct)
return self.app.render_currency(cstruct)
return super().serialize(field, cstruct, **kw)
class FileDownloadWidget(Widget): class FileDownloadWidget(Widget):
""" """
Widget for use with :class:`~wuttaweb.forms.schema.FileDownload` Widget for use with :class:`~wuttaweb.forms.schema.FileDownload`

View file

@ -32,7 +32,6 @@ from collections import namedtuple, OrderedDict
import sqlalchemy as sa import sqlalchemy as sa
from sqlalchemy import orm from sqlalchemy import orm
from sqlalchemy_utils import get_columns
import paginate import paginate
from paginate_sqlalchemy import SqlalchemyOrmPage from paginate_sqlalchemy import SqlalchemyOrmPage
@ -1117,16 +1116,19 @@ class Grid:
filters = filters or {} filters = filters or {}
if self.model_class: if self.model_class:
# nb. i first tried self.get_model_columns() but my notes # TODO: i tried using self.get_model_columns() here but in
# say that was too aggressive in many cases. then i tried # many cases that will be too aggressive. however it is
# using the *subset* of self.columns, just the ones which # often the case that the *grid* columns are a subset of
# corresponded to a property on the model class. and now # the unerlying *table* columns. so until a better way
# i am using sa-utils to give the "true" column list.. # is found, we choose "too few" instead of "too many"
for col in get_columns(self.model_class): # filters here. surely must improve it at some point.
if col.key in filters: for key in self.columns:
if key in filters:
continue continue
prop = getattr(self.model_class, col.key) prop = getattr(self.model_class, key, None)
filters[prop.key] = self.make_filter(prop) if (prop and hasattr(prop, 'property')
and isinstance(prop.property, orm.ColumnProperty)):
filters[prop.key] = self.make_filter(prop)
return filters return filters

View file

@ -92,9 +92,6 @@ class SessionProgress(ProgressBase):
""" """
def __init__(self, request, key, success_msg=None, success_url=None, error_url=None): def __init__(self, request, key, success_msg=None, success_url=None, error_url=None):
self.request = request
self.config = self.request.wutta_config
self.app = self.config.get_app()
self.key = key self.key = key
self.success_msg = success_msg self.success_msg = success_msg
self.success_url = success_url self.success_url = success_url
@ -140,7 +137,7 @@ class SessionProgress(ProgressBase):
""" """
self.session.load() self.session.load()
self.session['error'] = True self.session['error'] = True
self.session['error_msg'] = self.app.render_error(error) self.session['error_msg'] = str(error)
self.session['error_url'] = error_url or self.error_url self.session['error_url'] = error_url or self.error_url
self.session.save() self.session.save()

View file

@ -32,7 +32,6 @@ import uuid as _uuid
import warnings import warnings
import sqlalchemy as sa import sqlalchemy as sa
from sqlalchemy import orm
import colander import colander
from webhelpers2.html import HTML, tags from webhelpers2.html import HTML, tags
@ -479,36 +478,24 @@ def render_csrf_token(request, name='_csrf'):
return HTML.tag('div', tags.hidden(name, value=token, id=None), style='display:none;') return HTML.tag('div', tags.hidden(name, value=token, id=None), style='display:none;')
def get_model_fields(config, model_class, include_fk=False): def get_model_fields(config, model_class=None):
""" """
Convenience function to return a list of field names for the given Convenience function to return a list of field names for the given
:term:`data model` class. model class.
This logic only supports SQLAlchemy mapped classes and will use This logic only supports SQLAlchemy mapped classes and will use
that to determine the field listing if applicable. Otherwise this that to determine the field listing if applicable. Otherwise this
returns ``None``. returns ``None``.
:param config: App :term:`config object`.
:param model_class: Data model class.
:param include_fk: Whether to include foreign key column names in
the result. They are excluded by default, since the
relationship names are also included and generally preferred.
:returns: List of field names, or ``None`` if it could not be
determined.
""" """
if not model_class:
return
try: try:
mapper = sa.inspect(model_class) mapper = sa.inspect(model_class)
except sa.exc.NoInspectionAvailable: except sa.exc.NoInspectionAvailable:
return return
if include_fk: fields = [prop.key for prop in mapper.iterate_properties]
fields = [prop.key for prop in mapper.iterate_properties]
else:
fields = [prop.key for prop in mapper.iterate_properties
if not prop_is_fk(mapper, prop)]
# nb. we never want the continuum 'versions' prop # nb. we never want the continuum 'versions' prop
app = config.get_app() app = config.get_app()
@ -518,20 +505,6 @@ def get_model_fields(config, model_class, include_fk=False):
return fields return fields
def prop_is_fk(mapper, prop):
""" """
if not isinstance(prop, orm.ColumnProperty):
return False
prop_columns = [col.name for col in prop.columns]
for rel in mapper.relationships:
rel_columns = [col.name for col in rel.local_columns]
if rel_columns == prop_columns:
return True
return False
def make_json_safe(value, key=None, warn=True): def make_json_safe(value, key=None, warn=True):
""" """
Convert a Python value as needed, to ensure it is compatible with Convert a Python value as needed, to ensure it is compatible with

View file

@ -80,15 +80,6 @@ class TestWuttaEnum(WebTestCase):
self.assertIsInstance(widget, widgets.SelectWidget) self.assertIsInstance(widget, widgets.SelectWidget)
class TestWuttaMoney(WebTestCase):
def test_widget_maker(self):
enum = self.app.enum
typ = mod.WuttaMoney(self.request)
widget = typ.widget_maker()
self.assertIsInstance(widget, widgets.WuttaMoneyInputWidget)
class TestObjectRef(DataTestCase): class TestObjectRef(DataTestCase):
def setUp(self): def setUp(self):

View file

@ -1,7 +1,6 @@
# -*- coding: utf-8; -*- # -*- coding: utf-8; -*-
import datetime import datetime
import decimal
from unittest.mock import patch from unittest.mock import patch
import colander import colander
@ -108,36 +107,6 @@ class TestWuttaDateTimeWidget(WebTestCase):
self.assertEqual(result, '2024-12-12 13:49+0000') self.assertEqual(result, '2024-12-12 13:49+0000')
class TestWuttaMoneyInputWidget(WebTestCase):
def make_field(self, node, **kwargs):
# TODO: not sure why default renderer is in use even though
# pyramid_deform was included in setup? but this works..
kwargs.setdefault('renderer', deform.Form.default_renderer)
return deform.Field(node, **kwargs)
def make_widget(self, **kwargs):
return mod.WuttaMoneyInputWidget(self.request, **kwargs)
def test_serialize(self):
node = colander.SchemaNode(WuttaDateTime())
field = self.make_field(node)
widget = self.make_widget()
amount = decimal.Decimal('12.34')
# editable widget has normal text input
result = widget.serialize(field, str(amount))
self.assertIn('<b-input', result)
# readonly is rendered per app convention
result = widget.serialize(field, str(amount), readonly=True)
self.assertEqual(result, '$12.34')
# readonly w/ null value
result = widget.serialize(field, None, readonly=True)
self.assertEqual(result, '')
class TestFileDownloadWidget(WebTestCase): class TestFileDownloadWidget(WebTestCase):
def make_field(self, node, **kwargs): def make_field(self, node, **kwargs):

View file

@ -982,17 +982,6 @@ class TestGrid(WebTestCase):
self.assertEqual(filters['value'], 42) self.assertEqual(filters['value'], 42)
self.assertEqual(myfilters['value'], 42) self.assertEqual(myfilters['value'], 42)
# filters for all *true* columns by default, despite grid.columns
with patch.object(mod.Grid, 'make_filter'):
# nb. filters are MagicMock instances
grid = self.make_grid(model_class=model.User,
columns=['username', 'person'])
filters = grid.make_backend_filters()
self.assertIn('username', filters)
self.assertIn('active', filters)
# nb. relationship not included by default
self.assertNotIn('person', filters)
def test_make_filter(self): def test_make_filter(self):
model = self.app.model model = self.app.model

View file

@ -5,8 +5,6 @@ from unittest import TestCase
from pyramid import testing from pyramid import testing
from beaker.session import Session as BeakerSession from beaker.session import Session as BeakerSession
from wuttjamaican.testing import ConfigTestCase
from wuttaweb import progress as mod from wuttaweb import progress as mod
@ -33,11 +31,10 @@ class TestGetProgressSession(TestCase):
self.assertEqual(session.id, 'mockid.progress.foo') self.assertEqual(session.id, 'mockid.progress.foo')
class TestSessionProgress(ConfigTestCase): class TestSessionProgress(TestCase):
def setUp(self): def setUp(self):
self.setup_config() self.request = testing.DummyRequest()
self.request = testing.DummyRequest(wutta_config=self.config)
self.request.session.id = 'mockid' self.request.session.id = 'mockid'
def test_error_url(self): def test_error_url(self):

View file

@ -11,8 +11,6 @@ from fanstatic import Library, Resource
from pyramid import testing from pyramid import testing
from wuttjamaican.conf import WuttaConfig from wuttjamaican.conf import WuttaConfig
from wuttjamaican.testing import ConfigTestCase
from wuttaweb import util as mod from wuttaweb import util as mod
@ -465,10 +463,14 @@ class TestGetFormData(TestCase):
self.assertEqual(data, {'foo2': 'baz'}) self.assertEqual(data, {'foo2': 'baz'})
class TestGetModelFields(ConfigTestCase): class TestGetModelFields(TestCase):
def setUp(self):
self.config = WuttaConfig()
self.app = self.config.get_app()
def test_empty_model_class(self): def test_empty_model_class(self):
fields = mod.get_model_fields(self.config, None) fields = mod.get_model_fields(self.config)
self.assertIsNone(fields) self.assertIsNone(fields)
def test_unknown_model_class(self): def test_unknown_model_class(self):
@ -480,19 +482,6 @@ class TestGetModelFields(ConfigTestCase):
fields = mod.get_model_fields(self.config, model.Setting) fields = mod.get_model_fields(self.config, model.Setting)
self.assertEqual(fields, ['name', 'value']) self.assertEqual(fields, ['name', 'value'])
def test_include_fk(self):
model = self.app.model
# fk excluded by default
fields = mod.get_model_fields(self.config, model.User)
self.assertNotIn('person_uuid', fields)
self.assertIn('person', fields)
# fk can be included
fields = mod.get_model_fields(self.config, model.User, include_fk=True)
self.assertIn('person_uuid', fields)
self.assertIn('person', fields)
def test_avoid_versions(self): def test_avoid_versions(self):
model = self.app.model model = self.app.model