2014-12-03 11:07:21 -05:00
|
|
|
# -*- coding: utf-8 -*-
|
2015-01-06 08:21:39 -05:00
|
|
|
# Copyright 2014, 2015 OpenMarket Ltd
|
2014-12-03 11:07:21 -05:00
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
|
|
|
from synapse.api.constants import EventTypes
|
2014-12-05 11:20:48 -05:00
|
|
|
from . import EventBase
|
2014-12-03 11:07:21 -05:00
|
|
|
|
|
|
|
|
|
|
|
def prune_event(event):
|
|
|
|
""" Returns a pruned version of the given event, which removes all keys we
|
|
|
|
don't know about or think could potentially be dodgy.
|
|
|
|
|
|
|
|
This is used when we "redact" an event. We want to remove all fields that
|
|
|
|
the user has specified, but we do want to keep necessary information like
|
|
|
|
type, state_key etc.
|
|
|
|
"""
|
|
|
|
event_type = event.type
|
|
|
|
|
|
|
|
allowed_keys = [
|
|
|
|
"event_id",
|
|
|
|
"sender",
|
|
|
|
"room_id",
|
|
|
|
"hashes",
|
|
|
|
"signatures",
|
|
|
|
"content",
|
|
|
|
"type",
|
|
|
|
"state_key",
|
|
|
|
"depth",
|
|
|
|
"prev_events",
|
|
|
|
"prev_state",
|
|
|
|
"auth_events",
|
|
|
|
"origin",
|
|
|
|
"origin_server_ts",
|
2014-12-15 08:55:22 -05:00
|
|
|
"membership",
|
2014-12-03 11:07:21 -05:00
|
|
|
]
|
|
|
|
|
|
|
|
new_content = {}
|
|
|
|
|
|
|
|
def add_fields(*fields):
|
|
|
|
for field in fields:
|
|
|
|
if field in event.content:
|
|
|
|
new_content[field] = event.content[field]
|
|
|
|
|
|
|
|
if event_type == EventTypes.Member:
|
|
|
|
add_fields("membership")
|
|
|
|
elif event_type == EventTypes.Create:
|
|
|
|
add_fields("creator")
|
|
|
|
elif event_type == EventTypes.JoinRules:
|
|
|
|
add_fields("join_rule")
|
|
|
|
elif event_type == EventTypes.PowerLevels:
|
|
|
|
add_fields(
|
|
|
|
"users",
|
|
|
|
"users_default",
|
|
|
|
"events",
|
|
|
|
"events_default",
|
|
|
|
"events_default",
|
|
|
|
"state_default",
|
|
|
|
"ban",
|
|
|
|
"kick",
|
|
|
|
"redact",
|
|
|
|
)
|
|
|
|
elif event_type == EventTypes.Aliases:
|
|
|
|
add_fields("aliases")
|
|
|
|
|
|
|
|
allowed_fields = {
|
|
|
|
k: v
|
|
|
|
for k, v in event.get_dict().items()
|
|
|
|
if k in allowed_keys
|
|
|
|
}
|
|
|
|
|
|
|
|
allowed_fields["content"] = new_content
|
|
|
|
|
2014-12-11 08:25:19 -05:00
|
|
|
allowed_fields["unsigned"] = {}
|
|
|
|
|
|
|
|
if "age_ts" in event.unsigned:
|
|
|
|
allowed_fields["unsigned"]["age_ts"] = event.unsigned["age_ts"]
|
|
|
|
|
2014-12-03 11:07:21 -05:00
|
|
|
return type(event)(allowed_fields)
|
2014-12-05 11:20:48 -05:00
|
|
|
|
|
|
|
|
2015-01-26 11:11:28 -05:00
|
|
|
def serialize_event(e, time_now_ms, client_event=True):
|
2014-12-05 11:20:48 -05:00
|
|
|
# FIXME(erikj): To handle the case of presence events and the like
|
|
|
|
if not isinstance(e, EventBase):
|
|
|
|
return e
|
|
|
|
|
2015-01-26 11:11:28 -05:00
|
|
|
time_now_ms = int(time_now_ms)
|
|
|
|
|
2014-12-05 11:20:48 -05:00
|
|
|
# Should this strip out None's?
|
|
|
|
d = {k: v for k, v in e.get_dict().items()}
|
2015-01-08 09:27:04 -05:00
|
|
|
|
|
|
|
if not client_event:
|
|
|
|
# set the age and keep all other keys
|
|
|
|
if "age_ts" in d["unsigned"]:
|
2015-01-26 11:11:28 -05:00
|
|
|
d["unsigned"]["age"] = time_now_ms - d["unsigned"]["age_ts"]
|
2015-01-08 09:27:04 -05:00
|
|
|
return d
|
|
|
|
|
2014-12-05 11:20:48 -05:00
|
|
|
if "age_ts" in d["unsigned"]:
|
2015-01-26 11:11:28 -05:00
|
|
|
d["age"] = time_now_ms - d["unsigned"]["age_ts"]
|
2014-12-05 11:20:48 -05:00
|
|
|
del d["unsigned"]["age_ts"]
|
|
|
|
|
2014-12-08 04:08:26 -05:00
|
|
|
d["user_id"] = d.pop("sender", None)
|
|
|
|
|
2014-12-11 08:25:19 -05:00
|
|
|
if "redacted_because" in e.unsigned:
|
|
|
|
d["redacted_because"] = serialize_event(
|
2015-01-26 11:11:28 -05:00
|
|
|
e.unsigned["redacted_because"], time_now_ms
|
2014-12-11 08:25:19 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
del d["unsigned"]["redacted_because"]
|
|
|
|
|
|
|
|
if "redacted_by" in e.unsigned:
|
|
|
|
d["redacted_by"] = e.unsigned["redacted_by"]
|
|
|
|
del d["unsigned"]["redacted_by"]
|
|
|
|
|
2014-12-11 10:16:55 -05:00
|
|
|
if "replaces_state" in e.unsigned:
|
|
|
|
d["replaces_state"] = e.unsigned["replaces_state"]
|
|
|
|
del d["unsigned"]["replaces_state"]
|
|
|
|
|
|
|
|
if "prev_content" in e.unsigned:
|
|
|
|
d["prev_content"] = e.unsigned["prev_content"]
|
|
|
|
del d["unsigned"]["prev_content"]
|
|
|
|
|
2015-01-08 09:27:04 -05:00
|
|
|
del d["auth_events"]
|
|
|
|
del d["prev_events"]
|
|
|
|
del d["hashes"]
|
|
|
|
del d["signatures"]
|
|
|
|
d.pop("depth", None)
|
|
|
|
d.pop("unsigned", None)
|
|
|
|
d.pop("origin", None)
|
2014-12-11 08:25:19 -05:00
|
|
|
|
2014-12-09 05:58:31 -05:00
|
|
|
return d
|