sync.py
201 lines
| 7.0 KiB
| text/x-python
|
PythonLexer
neko259
|
r1177 | import xml.etree.ElementTree as et | ||
neko259
|
r1229 | from django.db import transaction | ||
neko259
|
r1239 | from boards.models import KeyPair, GlobalId, Signature, Post, Tag | ||
neko259
|
r1177 | |||
ENCODING_UNICODE = 'unicode' | ||||
TAG_MODEL = 'model' | ||||
TAG_REQUEST = 'request' | ||||
TAG_RESPONSE = 'response' | ||||
TAG_ID = 'id' | ||||
TAG_STATUS = 'status' | ||||
TAG_MODELS = 'models' | ||||
TAG_TITLE = 'title' | ||||
TAG_TEXT = 'text' | ||||
TAG_THREAD = 'thread' | ||||
TAG_PUB_TIME = 'pub-time' | ||||
TAG_SIGNATURES = 'signatures' | ||||
TAG_SIGNATURE = 'signature' | ||||
TAG_CONTENT = 'content' | ||||
TAG_ATTACHMENTS = 'attachments' | ||||
TAG_ATTACHMENT = 'attachment' | ||||
neko259
|
r1239 | TAG_TAGS = 'tags' | ||
TAG_TAG = 'tag' | ||||
neko259
|
r1177 | |||
TYPE_GET = 'get' | ||||
ATTR_VERSION = 'version' | ||||
ATTR_TYPE = 'type' | ||||
ATTR_NAME = 'name' | ||||
ATTR_VALUE = 'value' | ||||
ATTR_MIMETYPE = 'mimetype' | ||||
neko259
|
r1237 | ATTR_KEY = 'key' | ||
neko259
|
r1177 | |||
STATUS_SUCCESS = 'success' | ||||
class SyncManager: | ||||
neko259
|
r1236 | @staticmethod | ||
def generate_response_get(model_list: list): | ||||
neko259
|
r1177 | response = et.Element(TAG_RESPONSE) | ||
status = et.SubElement(response, TAG_STATUS) | ||||
status.text = STATUS_SUCCESS | ||||
models = et.SubElement(response, TAG_MODELS) | ||||
for post in model_list: | ||||
model = et.SubElement(models, TAG_MODEL) | ||||
model.set(ATTR_NAME, 'post') | ||||
content_tag = et.SubElement(model, TAG_CONTENT) | ||||
tag_id = et.SubElement(content_tag, TAG_ID) | ||||
post.global_id.to_xml_element(tag_id) | ||||
title = et.SubElement(content_tag, TAG_TITLE) | ||||
title.text = post.title | ||||
text = et.SubElement(content_tag, TAG_TEXT) | ||||
neko259
|
r1228 | text.text = post.get_sync_text() | ||
neko259
|
r1177 | |||
neko259
|
r1239 | thread = post.get_thread() | ||
if post.is_opening(): | ||||
tag_tags = et.SubElement(content_tag, TAG_TAGS) | ||||
for tag in thread.get_tags(): | ||||
tag_tag = et.SubElement(tag_tags, TAG_TAG) | ||||
tag_tag.text = tag.name | ||||
neko259
|
r1177 | else: | ||
neko259
|
r1239 | tag_thread = et.SubElement(content_tag, TAG_THREAD) | ||
thread_id = et.SubElement(tag_thread, TAG_ID) | ||||
thread.get_opening_post().global_id.to_xml_element(thread_id) | ||||
neko259
|
r1177 | |||
pub_time = et.SubElement(content_tag, TAG_PUB_TIME) | ||||
neko259
|
r1229 | pub_time.text = str(post.get_pub_time_str()) | ||
neko259
|
r1177 | |||
signatures_tag = et.SubElement(model, TAG_SIGNATURES) | ||||
neko259
|
r1242 | post_signatures = post.global_id.signature_set.all() | ||
neko259
|
r1177 | if post_signatures: | ||
neko259
|
r1238 | signatures = post_signatures | ||
# TODO Adding signature to a post is not yet added. For now this | ||||
# block is useless | ||||
neko259
|
r1177 | else: | ||
# TODO Maybe the signature can be computed only once after | ||||
# the post is added? Need to add some on_save signal queue | ||||
# and add this there. | ||||
key = KeyPair.objects.get(public_key=post.global_id.key) | ||||
neko259
|
r1386 | signature = Signature( | ||
neko259
|
r1177 | key_type=key.key_type, | ||
key=key.public_key, | ||||
neko259
|
r1504 | signature=key.sign(et.tostring(content_tag, encoding=ENCODING_UNICODE)), | ||
neko259
|
r1386 | global_id=post.global_id, | ||
) | ||||
signature.save() | ||||
signatures = [signature] | ||||
neko259
|
r1177 | for signature in signatures: | ||
signature_tag = et.SubElement(signatures_tag, TAG_SIGNATURE) | ||||
signature_tag.set(ATTR_TYPE, signature.key_type) | ||||
signature_tag.set(ATTR_VALUE, signature.signature) | ||||
neko259
|
r1237 | signature_tag.set(ATTR_KEY, signature.key) | ||
neko259
|
r1177 | |||
return et.tostring(response, ENCODING_UNICODE) | ||||
neko259
|
r1236 | @staticmethod | ||
neko259
|
r1229 | @transaction.atomic | ||
neko259
|
r1236 | def parse_response_get(response_xml): | ||
neko259
|
r1177 | tag_root = et.fromstring(response_xml) | ||
tag_status = tag_root.find(TAG_STATUS) | ||||
if STATUS_SUCCESS == tag_status.text: | ||||
tag_models = tag_root.find(TAG_MODELS) | ||||
for tag_model in tag_models: | ||||
tag_content = tag_model.find(TAG_CONTENT) | ||||
neko259
|
r1237 | |||
neko259
|
r1244 | signatures = SyncManager._verify_model(tag_content, tag_model) | ||
neko259
|
r1237 | |||
neko259
|
r1177 | tag_id = tag_content.find(TAG_ID) | ||
neko259
|
r1233 | global_id, exists = GlobalId.from_xml_element(tag_id) | ||
neko259
|
r1232 | |||
neko259
|
r1233 | if exists: | ||
print('Post with same ID already exists') | ||||
else: | ||||
global_id.save() | ||||
neko259
|
r1244 | for signature in signatures: | ||
signature.global_id = global_id | ||||
signature.save() | ||||
neko259
|
r1177 | |||
neko259
|
r1233 | title = tag_content.find(TAG_TITLE).text | ||
text = tag_content.find(TAG_TEXT).text | ||||
pub_time = tag_content.find(TAG_PUB_TIME).text | ||||
neko259
|
r1232 | |||
neko259
|
r1233 | thread = tag_content.find(TAG_THREAD) | ||
neko259
|
r1239 | tags = [] | ||
neko259
|
r1233 | if thread: | ||
neko259
|
r1386 | thread_id = thread.find(TAG_ID) | ||
op_global_id, exists = GlobalId.from_xml_element(thread_id) | ||||
if exists: | ||||
opening_post = Post.objects.get(global_id=op_global_id) | ||||
else: | ||||
raise Exception('Load the OP first') | ||||
neko259
|
r1233 | else: | ||
opening_post = None | ||||
neko259
|
r1239 | tag_tags = tag_content.find(TAG_TAGS) | ||
for tag_tag in tag_tags: | ||||
neko259
|
r1386 | tag, created = Tag.objects.get_or_create( | ||
name=tag_tag.text) | ||||
neko259
|
r1239 | tags.append(tag) | ||
neko259
|
r1177 | |||
neko259
|
r1233 | # TODO Check that the replied posts are already present | ||
# before adding new ones | ||||
neko259
|
r1229 | |||
neko259
|
r1238 | # TODO Get images | ||
neko259
|
r1233 | post = Post.objects.import_post( | ||
title=title, text=text, pub_time=pub_time, | ||||
neko259
|
r1244 | opening_post=opening_post, tags=tags, | ||
global_id=global_id) | ||||
neko259
|
r1177 | else: | ||
# TODO Throw an exception? | ||||
pass | ||||
neko259
|
r1237 | |||
@staticmethod | ||||
neko259
|
r1321 | def generate_response_pull(): | ||
response = et.Element(TAG_RESPONSE) | ||||
status = et.SubElement(response, TAG_STATUS) | ||||
status.text = STATUS_SUCCESS | ||||
models = et.SubElement(response, TAG_MODELS) | ||||
for post in Post.objects.all(): | ||||
tag_id = et.SubElement(models, TAG_ID) | ||||
post.global_id.to_xml_element(tag_id) | ||||
return et.tostring(response, ENCODING_UNICODE) | ||||
@staticmethod | ||||
neko259
|
r1238 | def _verify_model(tag_content, tag_model): | ||
neko259
|
r1237 | """ | ||
Verifies all signatures for a single model. | ||||
""" | ||||
neko259
|
r1244 | signatures = [] | ||
neko259
|
r1237 | |||
tag_signatures = tag_model.find(TAG_SIGNATURES) | ||||
for tag_signature in tag_signatures: | ||||
signature_type = tag_signature.get(ATTR_TYPE) | ||||
signature_value = tag_signature.get(ATTR_VALUE) | ||||
signature_key = tag_signature.get(ATTR_KEY) | ||||
neko259
|
r1244 | signature = Signature(key_type=signature_type, | ||
key=signature_key, | ||||
signature=signature_value) | ||||
neko259
|
r1386 | |||
content = et.tostring(tag_content, ENCODING_UNICODE) | ||||
neko259
|
r1244 | |||
neko259
|
r1237 | if not KeyPair.objects.verify( | ||
neko259
|
r1386 | signature, content): | ||
raise Exception('Invalid model signature for {}'.format(content)) | ||||
signatures.append(signature) | ||||
neko259
|
r1237 | |||
neko259
|
r1244 | return signatures | ||