如何以编程方式创建像 z3c.form 那样的详细事件?

How to programatically create a detailed event like z3c.form does?

我有一个简单的 event handler 用于查找实际更改的内容(已注册 IObjectModifiedEvent 事件),代码如下:

def on_change_do_something(obj, event):
    modified = False
    # check if the publication has changed
    for change in event.descriptions:
        if change.interface == IPublication:
            modified = True
            break

    if modified:
        # do something

所以我的问题是:如何以编程方式生成这些描述?我到处都在使用 plone.app.dexterity,所以 z3c.form 在使用表单时会自动执行此操作,但我想用单元测试对其进行测试。

我可能也误解了一些东西,但你可以简单地在你的代码中触发事件,使用相同的参数,如 z3c.form(类似于@keul 的评论)?

在 Plone 4.3.x 中进行了简短搜索后,我在 z3c.form.form 中找到了这个:

def applyChanges(self, data):
    content = self.getContent()
    changes = applyChanges(self, content, data)
    # ``changes`` is a dictionary; if empty, there were no changes
    if changes:
        # Construct change-descriptions for the object-modified event
        descriptions = []
        for interface, names in changes.items():
            descriptions.append(
                zope.lifecycleevent.Attributes(interface, *names))
        # Send out a detailed object-modified event
        zope.event.notify(
            zope.lifecycleevent.ObjectModifiedEvent(content, *descriptions))
    return changes

您需要两个测试用例,一个什么也不做,另一个通过您的代码。

applyChanges 在同一个模块 (z3c.form.form) 中,它遍历表单字段并计算包含所有更改的字典。

你应该在那里设置一个断点来检查字典是如何构建的。

之后你可以在你的测试用例中做同样的事情。

这样你就可以编写可读的测试用例。

def test_do_something_in_event(self)

    content = self.get_my_content()
    descriptions = self.get_event_descriptions()

    zope.event.notify(zope.lifecycleevent.ObjectModifiedEvent(content, *descriptions))

    self.assertSomething(...)        

恕我直言,将整个逻辑模拟掉对未来来说可能不是一个好主意,如果代码发生变化并且可能完全不同,您的测试仍然可以。

event.description 名义上是一个 IModificationDescription 对象,它本质上是一个 IAttributes 对象列表:每个 Attributes 对象都有一个接口(例如模式)和属性(例如字段名称列表)被修改。

最简单的解决方案是为每个更改的字段创建一个 zope.lifecycleevent.Attributes 对象,并将其作为参数传递给事件构造函数——示例:

# imports elided...

changelog = [
    Attributes(IFoo, 'some_fieldname_here'),
    Attributes(IMyBehaviorHere, 'some_behavior_provided_fieldname_here',
    ]
notify(ObjectModifiedEvent(context, *changelog)