summary refs log tree commit diff
diff options
context:
space:
mode:
authorErik Johnston <erik@matrix.org>2014-12-02 11:40:22 +0000
committerErik Johnston <erik@matrix.org>2014-12-02 11:40:22 +0000
commitc1e66800a937351614b2d19cdebbb141f786b03a (patch)
treea6d44d39738a92fadc2e1fe9ed834701283c1e25
parentChange DomainSpecificString so that it doesn't use a HomeServer object (diff)
downloadsynapse-c1e66800a937351614b2d19cdebbb141f786b03a.tar.xz
Begin fleshing out a new Event object
-rwxr-xr-xsetup.py1
-rw-r--r--synapse/events/__init__.py120
-rw-r--r--synapse/events/builder.py74
3 files changed, 195 insertions, 0 deletions
diff --git a/setup.py b/setup.py
index 9b38f790b9..d1b8f0680a 100755
--- a/setup.py
+++ b/setup.py
@@ -41,6 +41,7 @@ setup(
         "pynacl",
         "daemonize",
         "py-bcrypt",
+        "frozendict>=0.4",
     ],
     dependency_links=[
         "https://github.com/matrix-org/syutil/tarball/v0.0.2#egg=syutil-0.0.2",
diff --git a/synapse/events/__init__.py b/synapse/events/__init__.py
new file mode 100644
index 0000000000..eefc9d3b30
--- /dev/null
+++ b/synapse/events/__init__.py
@@ -0,0 +1,120 @@
+# -*- coding: utf-8 -*-
+# Copyright 2014 OpenMarket Ltd
+#
+# 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 frozendict import frozendict
+
+
+class _EventInternalMetadata(object):
+    def __init__(self, internal_metadata_dict):
+        self.__dict__ = internal_metadata_dict
+
+    def get_dict(self):
+        return dict(self.__dict__)
+
+
+class Event(object):
+    def __init__(self, event_dict, internal_metadata_dict={}):
+        self._signatures = event_dict.get("signatures", {})
+        self._unsigned = event_dict.get("unsigned", {})
+
+        self._original = {
+            k: v
+            for k, v in event_dict.items()
+            if k not in ["signatures", "unsigned"]
+        }
+
+        self._event_dict = frozendict(self._original)
+
+        self.internal_metadata = _EventInternalMetadata(
+            internal_metadata_dict
+        )
+
+    @property
+    def auth_events(self):
+        return self._event_dict["auth_events"]
+
+    @property
+    def content(self):
+        return self._event_dict["content"]
+
+    @property
+    def event_id(self):
+        return self._event_dict["event_id"]
+
+    @property
+    def hashes(self):
+        return self._event_dict["hashes"]
+
+    @property
+    def origin(self):
+        return self._event_dict["origin"]
+
+    @property
+    def prev_events(self):
+        return self._event_dict["prev_events"]
+
+    @property
+    def prev_state(self):
+        return self._event_dict["prev_state"]
+
+    @property
+    def room_id(self):
+        return self._event_dict["room_id"]
+
+    @property
+    def signatures(self):
+        return self._signatures
+
+    @property
+    def state_key(self):
+        return self._event_dict["state_key"]
+
+    @property
+    def type(self):
+        return self._event_dict["type"]
+
+    @property
+    def unsigned(self):
+        return self._unsigned
+
+    @property
+    def user_id(self):
+        return self._event_dict["sender"]
+
+    @property
+    def sender(self):
+        return self._event_dict["sender"]
+
+    def get_dict(self):
+        d = dict(self._original)
+        d.update({
+            "signatures": self._signatures,
+            "unsigned": self._unsigned,
+        })
+
+        return d
+
+    def get_internal_metadata_dict(self):
+        return self.internal_metadata.get_dict()
+
+    def get_pdu_json(self, time_now=None):
+        pdu_json = self.get_dict()
+
+        if time_now is not None and "age_ts" in pdu_json["unsigned"]:
+            age = time_now - pdu_json["unsigned"]["age_ts"]
+            pdu_json.setdefault("unsigned", {})["age"] = int(age)
+            del pdu_json["unsigned"]["age_ts"]
+
+        return pdu_json
\ No newline at end of file
diff --git a/synapse/events/builder.py b/synapse/events/builder.py
new file mode 100644
index 0000000000..d741795bc5
--- /dev/null
+++ b/synapse/events/builder.py
@@ -0,0 +1,74 @@
+# -*- coding: utf-8 -*-
+# Copyright 2014 OpenMarket Ltd
+#
+# 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 . import Event
+
+from synapse.types import EventID
+
+from synapse.util.stringutils import random_string
+
+
+class EventBuilder(object):
+    def __init__(self, key_values={}):
+        self._event_dict = dict(key_values)
+        self._metadata = {}
+
+    def update_event_key(self, key, value):
+        self._event_dict[key] = value
+
+    def update_event_keys(self, other_dict):
+        self._event_dict.update(other_dict)
+
+    def update_internal_key(self, key, value):
+        self._metadata[key] = value
+
+    def build(self):
+        return Event(
+            self._event_dict,
+            self._metadata,
+        )
+
+
+class EventBuilderFactory(object):
+    def __init__(self, clock, hostname):
+        self.clock = clock
+        self.hostname = hostname
+
+        self.event_id_count = 0
+
+    def create_event_id(self):
+        i = str(self.event_id_count)
+        self.event_id_count += 1
+
+        local_part = str(int(self.clock.time())) + i + random_string(5)
+
+        e_id = EventID.create(local_part, self.hostname)
+
+        return e_id.to_string()
+
+    def new(self, key_values={}):
+        if "event_id" not in key_values:
+            key_values["event_id"] = self.create_event_id()
+
+        time_now = self.clock.time_msec()
+
+        key_values.setdefault("origin", self.hostname)
+        key_values.setdefault("origin_server_ts", time_now)
+
+        if "unsigned" in key_values:
+            age = key_values["unsigned"].pop("age", 0)
+            key_values["unsigned"].setdefault("age_ts", time_now - age)
+
+        return EventBuilder(key_values=key_values,)
\ No newline at end of file