2024-12-13 22:20:04 -06:00
|
|
|
# -*- coding: utf-8; -*-
|
|
|
|
|
|
|
|
import datetime
|
|
|
|
from unittest.mock import patch, MagicMock
|
|
|
|
|
2024-12-14 10:38:49 -06:00
|
|
|
from sqlalchemy import orm
|
2024-12-13 22:20:04 -06:00
|
|
|
from pyramid.httpexceptions import HTTPFound
|
|
|
|
|
|
|
|
from wuttjamaican.db import model
|
|
|
|
from wuttjamaican.batch import BatchHandler
|
2024-12-14 20:42:57 -06:00
|
|
|
from wuttaweb.views import MasterView, batch as mod
|
2024-12-13 22:20:04 -06:00
|
|
|
from wuttaweb.progress import SessionProgress
|
2025-01-06 16:47:48 -06:00
|
|
|
from wuttaweb.testing import WebTestCase
|
2024-12-13 22:20:04 -06:00
|
|
|
|
|
|
|
|
|
|
|
class MockBatch(model.BatchMixin, model.Base):
|
|
|
|
__tablename__ = 'testing_batch_mock'
|
|
|
|
|
|
|
|
class MockBatchRow(model.BatchRowMixin, model.Base):
|
|
|
|
__tablename__ = 'testing_batch_mock_row'
|
|
|
|
__batch_class__ = MockBatch
|
|
|
|
|
2024-12-14 10:38:49 -06:00
|
|
|
MockBatch.__row_class__ = MockBatchRow
|
|
|
|
|
2024-12-13 22:20:04 -06:00
|
|
|
class MockBatchHandler(BatchHandler):
|
|
|
|
model_class = MockBatch
|
|
|
|
|
|
|
|
|
|
|
|
class TestBatchMasterView(WebTestCase):
|
|
|
|
|
2024-12-14 10:38:49 -06:00
|
|
|
def setUp(self):
|
|
|
|
self.setup_web()
|
|
|
|
|
|
|
|
# nb. create MockBatch, MockBatchRow
|
|
|
|
model.Base.metadata.create_all(bind=self.session.bind)
|
|
|
|
|
2024-12-14 20:42:57 -06:00
|
|
|
def make_handler(self):
|
|
|
|
return MockBatchHandler(self.config)
|
|
|
|
|
2024-12-14 10:38:49 -06:00
|
|
|
def make_view(self):
|
|
|
|
return mod.BatchMasterView(self.request)
|
|
|
|
|
2024-12-13 22:20:04 -06:00
|
|
|
def test_get_batch_handler(self):
|
|
|
|
self.assertRaises(NotImplementedError, mod.BatchMasterView, self.request)
|
|
|
|
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=42):
|
|
|
|
view = mod.BatchMasterView(self.request)
|
|
|
|
self.assertEqual(view.batch_handler, 42)
|
|
|
|
|
2024-12-14 20:42:57 -06:00
|
|
|
def test_get_fallback_templates(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
view = self.make_view()
|
|
|
|
templates = view.get_fallback_templates('view')
|
|
|
|
self.assertEqual(templates, [
|
|
|
|
'/batch/view.mako',
|
|
|
|
'/master/view.mako',
|
|
|
|
])
|
|
|
|
|
|
|
|
def test_render_to_response(self):
|
|
|
|
model = self.app.model
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
|
|
|
|
user = model.User(username='barney')
|
|
|
|
self.session.add(user)
|
|
|
|
batch = handler.make_batch(self.session, created_by=user)
|
|
|
|
self.session.add(batch)
|
|
|
|
self.session.flush()
|
|
|
|
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
with patch.object(MasterView, 'render_to_response') as render_to_response:
|
|
|
|
view = self.make_view()
|
|
|
|
response = view.render_to_response('view', {'instance': batch})
|
|
|
|
self.assertTrue(render_to_response.called)
|
|
|
|
context = render_to_response.call_args[0][1]
|
|
|
|
self.assertIs(context['batch'], batch)
|
|
|
|
self.assertIs(context['batch_handler'], handler)
|
|
|
|
|
2024-12-13 22:20:04 -06:00
|
|
|
def test_configure_grid(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.multiple(mod.BatchMasterView, create=True, model_class=MockBatch):
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
view = mod.BatchMasterView(self.request)
|
|
|
|
grid = view.make_model_grid()
|
|
|
|
# nb. coverage only; tests nothing
|
|
|
|
view.configure_grid(grid)
|
|
|
|
|
|
|
|
def test_render_batch_id(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
view = mod.BatchMasterView(self.request)
|
|
|
|
batch = MockBatch(id=42)
|
|
|
|
|
|
|
|
result = view.render_batch_id(batch, 'id', 42)
|
|
|
|
self.assertEqual(result, '00000042')
|
|
|
|
|
|
|
|
result = view.render_batch_id(batch, 'id', None)
|
|
|
|
self.assertIsNone(result)
|
|
|
|
|
|
|
|
def test_get_instance_title(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
view = mod.BatchMasterView(self.request)
|
|
|
|
|
|
|
|
batch = MockBatch(id=42)
|
|
|
|
result = view.get_instance_title(batch)
|
|
|
|
self.assertEqual(result, "00000042")
|
|
|
|
|
|
|
|
batch = MockBatch(id=43, description="runnin some numbers")
|
|
|
|
result = view.get_instance_title(batch)
|
|
|
|
self.assertEqual(result, "00000043 runnin some numbers")
|
|
|
|
|
|
|
|
def test_configure_form(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.multiple(mod.BatchMasterView, create=True, model_class=MockBatch):
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
view = mod.BatchMasterView(self.request)
|
|
|
|
|
|
|
|
# creating
|
|
|
|
with patch.object(view, 'creating', new=True):
|
|
|
|
form = view.make_model_form(model_instance=None)
|
|
|
|
view.configure_form(form)
|
|
|
|
|
|
|
|
batch = MockBatch(id=42)
|
|
|
|
|
|
|
|
# viewing
|
|
|
|
with patch.object(view, 'viewing', new=True):
|
|
|
|
form = view.make_model_form(model_instance=batch)
|
|
|
|
view.configure_form(form)
|
|
|
|
|
|
|
|
# editing
|
|
|
|
with patch.object(view, 'editing', new=True):
|
|
|
|
form = view.make_model_form(model_instance=batch)
|
|
|
|
view.configure_form(form)
|
|
|
|
|
|
|
|
# deleting
|
|
|
|
with patch.object(view, 'deleting', new=True):
|
|
|
|
form = view.make_model_form(model_instance=batch)
|
|
|
|
view.configure_form(form)
|
|
|
|
|
|
|
|
# viewing (executed)
|
|
|
|
batch.executed = datetime.datetime.now()
|
|
|
|
with patch.object(view, 'viewing', new=True):
|
|
|
|
form = view.make_model_form(model_instance=batch)
|
|
|
|
view.configure_form(form)
|
|
|
|
|
|
|
|
def test_objectify(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.multiple(mod.BatchMasterView, create=True, model_class=MockBatch):
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
with patch.object(mod.BatchMasterView, 'Session', return_value=self.session):
|
|
|
|
view = mod.BatchMasterView(self.request)
|
|
|
|
|
|
|
|
# create batch
|
|
|
|
with patch.object(view, 'creating', new=True):
|
|
|
|
form = view.make_model_form(model_instance=None)
|
|
|
|
form.validated = {}
|
|
|
|
batch = view.objectify(form)
|
|
|
|
self.assertIsInstance(batch.id, int)
|
|
|
|
self.assertTrue(batch.id > 0)
|
|
|
|
|
|
|
|
# edit batch
|
|
|
|
with patch.object(view, 'editing', new=True):
|
|
|
|
with patch.object(view.batch_handler, 'make_batch') as make_batch:
|
|
|
|
form = view.make_model_form(model_instance=batch)
|
|
|
|
form.validated = {'description': 'foo'}
|
|
|
|
self.assertIsNone(batch.description)
|
|
|
|
batch = view.objectify(form)
|
|
|
|
self.assertEqual(batch.description, 'foo')
|
|
|
|
|
|
|
|
def test_redirect_after_create(self):
|
|
|
|
self.pyramid_config.add_route('mock_batches.view', '/batch/mock/{uuid}')
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
with patch.multiple(mod.BatchMasterView, create=True,
|
|
|
|
model_class=MockBatch,
|
|
|
|
route_prefix='mock_batches'):
|
|
|
|
view = mod.BatchMasterView(self.request)
|
|
|
|
batch = MockBatch(id=42)
|
|
|
|
|
|
|
|
# typically redirect to view batch
|
|
|
|
result = view.redirect_after_create(batch)
|
|
|
|
self.assertIsInstance(result, HTTPFound)
|
|
|
|
|
|
|
|
# unless populating in which case thread is launched
|
|
|
|
self.request.session.id = 'abcdefghijk'
|
|
|
|
with patch.object(mod, 'threading') as threading:
|
|
|
|
thread = MagicMock()
|
|
|
|
threading.Thread.return_value = thread
|
|
|
|
with patch.object(view.batch_handler, 'should_populate', return_value=True):
|
|
|
|
with patch.object(view, 'render_progress') as render_progress:
|
|
|
|
view.redirect_after_create(batch)
|
|
|
|
self.assertTrue(threading.Thread.called)
|
|
|
|
thread.start.assert_called_once_with()
|
|
|
|
self.assertTrue(render_progress.called)
|
|
|
|
|
2024-12-14 20:42:57 -06:00
|
|
|
def test_delete_instance(self):
|
|
|
|
model = self.app.model
|
|
|
|
handler = self.make_handler()
|
|
|
|
|
|
|
|
user = model.User(username='barney')
|
|
|
|
self.session.add(user)
|
|
|
|
|
|
|
|
batch = handler.make_batch(self.session, created_by=user)
|
|
|
|
self.session.add(batch)
|
|
|
|
self.session.flush()
|
|
|
|
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
view = self.make_view()
|
|
|
|
|
|
|
|
self.assertEqual(self.session.query(MockBatch).count(), 1)
|
|
|
|
view.delete_instance(batch)
|
|
|
|
self.assertEqual(self.session.query(MockBatch).count(), 0)
|
|
|
|
|
2024-12-13 22:20:04 -06:00
|
|
|
def test_populate_thread(self):
|
|
|
|
model = self.app.model
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
with patch.multiple(mod.BatchMasterView, create=True, model_class=MockBatch):
|
|
|
|
view = mod.BatchMasterView(self.request)
|
|
|
|
user = model.User(username='barney')
|
|
|
|
self.session.add(user)
|
|
|
|
batch = MockBatch(id=42, created_by=user)
|
|
|
|
self.session.add(batch)
|
|
|
|
self.session.commit()
|
|
|
|
|
|
|
|
# nb. use our session within thread method
|
|
|
|
with patch.object(self.app, 'make_session', return_value=self.session):
|
|
|
|
|
|
|
|
# nb. prevent closing our session
|
|
|
|
with patch.object(self.session, 'close') as close:
|
|
|
|
|
|
|
|
# without progress
|
|
|
|
view.populate_thread(batch.uuid)
|
|
|
|
close.assert_called_once_with()
|
|
|
|
close.reset_mock()
|
|
|
|
|
|
|
|
# with progress
|
|
|
|
self.request.session.id = 'abcdefghijk'
|
|
|
|
view.populate_thread(batch.uuid,
|
|
|
|
progress=SessionProgress(self.request,
|
|
|
|
'populate_mock_batch'))
|
|
|
|
close.assert_called_once_with()
|
|
|
|
close.reset_mock()
|
|
|
|
|
|
|
|
# failure to populate, without progress
|
|
|
|
with patch.object(view.batch_handler, 'do_populate', side_effect=RuntimeError):
|
|
|
|
view.populate_thread(batch.uuid)
|
|
|
|
close.assert_called_once_with()
|
|
|
|
close.reset_mock()
|
|
|
|
|
|
|
|
# failure to populate, with progress
|
|
|
|
with patch.object(view.batch_handler, 'do_populate', side_effect=RuntimeError):
|
|
|
|
view.populate_thread(batch.uuid,
|
|
|
|
progress=SessionProgress(self.request,
|
|
|
|
'populate_mock_batch'))
|
|
|
|
close.assert_called_once_with()
|
|
|
|
close.reset_mock()
|
|
|
|
|
|
|
|
# failure for batch to appear
|
|
|
|
self.session.delete(batch)
|
|
|
|
self.session.commit()
|
|
|
|
# nb. should give up waiting after 1 second
|
|
|
|
self.assertRaises(RuntimeError, view.populate_thread, batch.uuid)
|
2024-12-14 10:38:49 -06:00
|
|
|
|
2024-12-14 20:42:57 -06:00
|
|
|
def test_execute(self):
|
|
|
|
self.pyramid_config.add_route('mock_batches.view', '/batch/mock/{uuid}')
|
|
|
|
model = self.app.model
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
|
|
|
|
user = model.User(username='barney')
|
|
|
|
self.session.add(user)
|
|
|
|
batch = handler.make_batch(self.session, created_by=user)
|
|
|
|
self.session.add(batch)
|
|
|
|
self.session.commit()
|
|
|
|
|
|
|
|
with patch.multiple(mod.BatchMasterView, create=True,
|
|
|
|
model_class=MockBatch,
|
|
|
|
route_prefix='mock_batches',
|
|
|
|
get_batch_handler=MagicMock(return_value=handler),
|
|
|
|
get_instance=MagicMock(return_value=batch)):
|
|
|
|
view = self.make_view()
|
|
|
|
|
|
|
|
# batch executes okay
|
|
|
|
response = view.execute()
|
|
|
|
self.assertEqual(response.status_code, 302) # redirect to "view batch"
|
|
|
|
self.assertFalse(self.request.session.peek_flash('error'))
|
|
|
|
|
|
|
|
# but cannot be executed again
|
|
|
|
response = view.execute()
|
|
|
|
self.assertEqual(response.status_code, 302) # redirect to "view batch"
|
|
|
|
# nb. flash has error this time
|
|
|
|
self.assertTrue(self.request.session.peek_flash('error'))
|
|
|
|
|
2024-12-14 10:38:49 -06:00
|
|
|
def test_get_row_model_class(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
view = self.make_view()
|
|
|
|
|
|
|
|
self.assertRaises(AttributeError, view.get_row_model_class)
|
|
|
|
|
|
|
|
# row class determined from batch class
|
|
|
|
with patch.object(mod.BatchMasterView, 'model_class', new=MockBatch, create=True):
|
|
|
|
cls = view.get_row_model_class()
|
|
|
|
self.assertIs(cls, MockBatchRow)
|
|
|
|
|
|
|
|
self.assertRaises(AttributeError, view.get_row_model_class)
|
|
|
|
|
|
|
|
# view may specify row class
|
|
|
|
with patch.object(mod.BatchMasterView, 'row_model_class', new=MockBatchRow, create=True):
|
|
|
|
cls = view.get_row_model_class()
|
|
|
|
self.assertIs(cls, MockBatchRow)
|
|
|
|
|
|
|
|
def test_get_row_grid_data(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
model = self.app.model
|
|
|
|
|
|
|
|
user = model.User(username='barney')
|
|
|
|
self.session.add(user)
|
|
|
|
|
|
|
|
batch = handler.make_batch(self.session, created_by=user)
|
|
|
|
self.session.add(batch)
|
|
|
|
row = handler.make_row()
|
|
|
|
handler.add_row(batch, row)
|
|
|
|
self.session.flush()
|
|
|
|
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
|
|
|
|
view = self.make_view()
|
|
|
|
self.assertRaises(AttributeError, view.get_row_grid_data, batch)
|
|
|
|
|
|
|
|
Session = MagicMock(return_value=self.session)
|
|
|
|
Session.query.side_effect = lambda m: self.session.query(m)
|
|
|
|
with patch.multiple(mod.BatchMasterView, create=True,
|
|
|
|
Session=Session,
|
|
|
|
model_class=MockBatch):
|
|
|
|
|
|
|
|
view = self.make_view()
|
|
|
|
data = view.get_row_grid_data(batch)
|
|
|
|
self.assertIsInstance(data, orm.Query)
|
|
|
|
self.assertEqual(data.count(), 1)
|
|
|
|
|
|
|
|
def test_configure_row_grid(self):
|
|
|
|
handler = MockBatchHandler(self.config)
|
|
|
|
model = self.app.model
|
|
|
|
|
|
|
|
user = model.User(username='barney')
|
|
|
|
self.session.add(user)
|
|
|
|
|
|
|
|
batch = handler.make_batch(self.session, created_by=user)
|
|
|
|
self.session.add(batch)
|
|
|
|
row = handler.make_row()
|
|
|
|
handler.add_row(batch, row)
|
|
|
|
self.session.flush()
|
|
|
|
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=handler):
|
|
|
|
|
|
|
|
Session = MagicMock(return_value=self.session)
|
|
|
|
Session.query.side_effect = lambda m: self.session.query(m)
|
|
|
|
with patch.multiple(mod.BatchMasterView, create=True,
|
|
|
|
Session=Session,
|
|
|
|
model_class=MockBatch):
|
|
|
|
|
|
|
|
with patch.object(self.request, 'matchdict', new={'uuid': batch.uuid}):
|
|
|
|
view = self.make_view()
|
|
|
|
grid = view.make_row_model_grid(batch)
|
|
|
|
self.assertIn('sequence', grid.labels)
|
|
|
|
self.assertEqual(grid.labels['sequence'], "Seq.")
|
2024-12-14 20:42:57 -06:00
|
|
|
|
2025-01-02 23:14:10 -06:00
|
|
|
def test_render_row_status(self):
|
|
|
|
with patch.object(mod.BatchMasterView, 'get_batch_handler', return_value=None):
|
|
|
|
view = self.make_view()
|
|
|
|
row = MagicMock(foo=1, STATUS={1: 'bar'})
|
|
|
|
self.assertEqual(view.render_row_status(row, 'foo', 1), 'bar')
|
|
|
|
|
2024-12-14 20:42:57 -06:00
|
|
|
def test_defaults(self):
|
|
|
|
# nb. coverage only
|
|
|
|
with patch.object(mod.BatchMasterView, 'model_class', new=MockBatch, create=True):
|
|
|
|
mod.BatchMasterView.defaults(self.pyramid_config)
|