带有Python的Apache Beam中的会话窗口

时间:2019-03-20 13:32:57

标签: python apache-beam

我有一系列的用户事件。我已经将它们映射到KV {userId,event},并分配了时间戳。

这将以流模式运行。我希望能够创建以下输入输出结果:

会话窗口间隔= 1

  • 输入:user=1, timestamp=1, event=a
  • 输入:user=2, timestamp=2, event=a
  • 输入:user=2, timestamp=3, event=a
  • 输入:user=1, timestamp=2, event=b
  • 时间:lwm=3
  • 输出:user=1, [ { event=a, timestamp=1 }, { event=b, timestamp=2 } ]
  • 时间:lwm=4
  • 输出:user=2, [ { event=a, timestamp=2 }, { event=a, timestamp=3 } ]

这样我就可以编写函数以减少用户在会话窗口中的事件列表以及会话窗口的开始和结束时间。

我该怎么写? (如果回答,请“查看示例”,这不是一个有效的答案,因为它们从不将事件列表以窗口作为参数输入到reducer中)

1 个答案:

答案 0 :(得分:0)

如果我正确理解这一点,那么这将是对question的后续行动,自然可以通过在我的解决方案中建议的那样添加Group By Key步骤来实现。

因此,请参阅我以前的解释,如果我们有这样的管道,则仅关注更改:

events = (p
  | 'Create Events' >> beam.Create(user1_data + user2_data) \
  | 'Add Timestamps' >> beam.Map(lambda x: beam.window.TimestampedValue(x, x['timestamp'])) \
  | 'keyed_on_user_id'      >> beam.Map(lambda x: (x['user_id'], x))
  | 'user_session_window'   >> beam.WindowInto(window.Sessions(session_gap),
                                             timestamp_combiner=window.TimestampCombiner.OUTPUT_AT_EOW) \
  | 'Group' >> beam.GroupByKey() \
  | 'analyze_session'         >> beam.ParDo(AnalyzeSession()))

现在,元素按照您在问题描述中的描述进行排列,因此我们可以简单地将它们记录在AnalyzeSession中:

class AnalyzeSession(beam.DoFn):
  """Prints per session information"""
  def process(self, element, window=beam.DoFn.WindowParam):
    logging.info(element)
    yield element

获得所需的结果:

INFO:root:('Groot', [{'timestamp': 1554203778.904401, 'user_id': 'Groot', 'value': 'event_0'}, {'timestamp': 1554203780.904401, 'user_id': 'Groot', 'value': 'event_1'}])
INFO:root:('Groot', [{'timestamp': 1554203786.904402, 'user_id': 'Groot', 'value': 'event_2'}])
INFO:root:('Thanos', [{'timestamp': 1554203792.904399, 'user_id': 'Thanos', 'value': 'event_4'}])
INFO:root:('Thanos', [{'timestamp': 1554203784.904398, 'user_id': 'Thanos', 'value': 'event_3'}, {'timestamp': 1554203777.904395, 'user_id': 'Thanos', 'value': 'event_0'}, {'timestamp': 1554203778.904397, 'user_id': 'Thanos', 'value': 'event_1'}, {'timestamp': 1554203780.904398, 'user_id': 'Thanos', 'value': 'event_2'}])

如果要避免使用多余的信息,例如将user_idtimestamp作为值的一部分,则可以在Map步骤中将其删除。 根据完整的用例(即减少每个会话级别的汇总事件),我们可以执行类似以下操作的事情:计算事件数或会话持续时间:

class AnalyzeSession(beam.DoFn):
  """Prints per session information"""
  def process(self, element, window=beam.DoFn.WindowParam):
    user = element[0]
    num_events = str(len(element[1]))
    window_end = window.end.to_utc_datetime()
    window_start = window.start.to_utc_datetime()
    session_duration = window_end - window_start

    logging.info(">>> User %s had %s event(s) in %s session", user, num_events, session_duration)

    yield element

对于我的示例,将输出以下内容:

INFO:root:>>> User Groot had 2 event(s) in 0:00:07 session
INFO:root:>>> User Groot had 1 event(s) in 0:00:05 session
INFO:root:>>> User Thanos had 4 event(s) in 0:00:12 session
INFO:root:>>> User Thanos had 1 event(s) in 0:00:05 session

完整代码here