##// END OF EJS Templates
Use update-time of a post instead of version
neko259 -
r1928:f3702378 default
parent child Browse files
Show More
@@ -0,0 +1,19
1 # -*- coding: utf-8 -*-
2 # Generated by Django 1.11 on 2017-09-27 12:38
3 from __future__ import unicode_literals
4
5 from django.db import migrations
6
7
8 class Migration(migrations.Migration):
9
10 dependencies = [
11 ('boards', '0063_auto_20170301_1058'),
12 ]
13
14 operations = [
15 migrations.RemoveField(
16 model_name='post',
17 name='version',
18 ),
19 ]
@@ -16,7 +16,7 class PostAdmin(admin.ModelAdmin):
16 16 exclude = ('referenced_posts', 'refmap', 'images', 'global_id')
17 17 readonly_fields = ('poster_ip', 'thread', 'linked_images',
18 18 'attachments', 'uid', 'url', 'pub_time', 'opening', 'linked_global_id',
19 'version', 'foreign', 'tags')
19 'foreign', 'tags')
20 20
21 21 def ban_poster(self, request, queryset):
22 22 bans = 0
@@ -63,7 +63,6 class PostAdmin(admin.ModelAdmin):
63 63 return ', '.join([tag.get_name() for tag in obj.get_tags()])
64 64
65 65 def save_model(self, request, obj, form, change):
66 obj.increment_version()
67 66 obj.save()
68 67 obj.clear_cache()
69 68
@@ -127,8 +126,6 class ThreadAdmin(admin.ModelAdmin):
127 126
128 127 def save_model(self, request, obj, form, change):
129 128 op = obj.get_opening_post()
130 op.increment_version()
131 op.save(update_fields=['version'])
132 129 obj.save()
133 130 op.clear_cache()
134 131
@@ -4,9 +4,10 import xml.etree.ElementTree as ET
4 4
5 5 import httplib2
6 6 from django.core.management import BaseCommand
7 from django.utils.dateparse import parse_datetime
7 8
8 9 from boards.models import GlobalId
9 from boards.models.post.sync import SyncManager, TAG_ID, TAG_VERSION
10 from boards.models.post.sync import SyncManager, TAG_ID, TAG_UPDATE_TIME
10 11
11 12 __author__ = 'neko259'
12 13
@@ -85,12 +86,12 class Command(BaseCommand):
85 86 for model in models:
86 87 tag_id = model.find(TAG_ID)
87 88 global_id, exists = GlobalId.from_xml_element(tag_id)
88 tag_version = model.find(TAG_VERSION)
89 if tag_version is not None:
90 version = int(tag_version.text) or 1
89 tag_update_time = model.find(TAG_UPDATE_TIME)
90 if tag_update_time:
91 update_time = tag_update_time.text
91 92 else:
92 version = 1
93 if not exists or global_id.post.version < version:
93 update_time = None
94 if not exists or update_time is None or global_id.post.last_edit_time < parse_datetime(update_time):
94 95 logger.debug('Processed (+) post {}'.format(global_id))
95 96 ids_to_sync.append(global_id)
96 97 else:
@@ -100,7 +100,6 class Post(models.Model, Viewable):
100 100 tripcode = models.CharField(max_length=50, blank=True, default='')
101 101 opening = models.BooleanField(db_index=True)
102 102 hidden = models.BooleanField(default=False)
103 version = models.IntegerField(default=1)
104 103
105 104 def __str__(self):
106 105 return 'P#{}/{}'.format(self.id, self.get_title())
@@ -341,9 +340,6 class Post(models.Model, Viewable):
341 340 def set_hidden(self, hidden):
342 341 self.hidden = hidden
343 342
344 def increment_version(self):
345 self.version = F('version') + 1
346
347 343 def clear_cache(self):
348 344 """
349 345 Clears sync data (content cache, signatures etc).
@@ -138,7 +138,7 class PostManager(models.Manager):
138 138 @transaction.atomic
139 139 def import_post(self, title: str, text: str, pub_time: str, global_id,
140 140 opening_post=None, tags=list(), files=list(),
141 file_urls=list(), tripcode=None, version=1):
141 file_urls=list(), tripcode=None, last_edit_time=None):
142 142 is_opening = opening_post is None
143 143 if is_opening:
144 144 thread = boards.models.thread.Thread.objects.create(
@@ -151,12 +151,11 class PostManager(models.Manager):
151 151 text=text,
152 152 pub_time=pub_time,
153 153 poster_ip=NO_IP,
154 last_edit_time=pub_time,
154 last_edit_time=last_edit_time or pub_time,
155 155 global_id=global_id,
156 156 opening=is_opening,
157 157 thread=thread,
158 tripcode=tripcode,
159 version=version)
158 tripcode=tripcode)
160 159
161 160 for file in files:
162 161 self._add_file_to_post(file, post)
@@ -170,12 +169,11 class PostManager(models.Manager):
170 169
171 170 @transaction.atomic
172 171 def update_post(self, post, title: str, text: str, pub_time: str,
173 tags=list(), files=list(), file_urls=list(), tripcode=None, version=1):
172 tags=list(), files=list(), file_urls=list(), tripcode=None):
174 173 post.title = title
175 174 post.text = text
176 175 post.pub_time = pub_time
177 176 post.tripcode = tripcode
178 post.version = version
179 177 post.save()
180 178
181 179 post.clear_cache()
@@ -1,6 +1,8
1 import logging
1 2 import xml.etree.ElementTree as et
2 import logging
3 from xml.etree import ElementTree
3
4 from django.db import transaction
5 from django.utils.dateparse import parse_datetime
4 6
5 7 from boards.abstracts.exceptions import SyncException
6 8 from boards.abstracts.sync_filters import ThreadFilter, TagsFilter,\
@@ -10,7 +12,6 from boards.models.attachment.downloader
10 12 from boards.models.signature import TAG_REQUEST, ATTR_TYPE, TYPE_GET, \
11 13 ATTR_VERSION, TAG_MODEL, ATTR_NAME, TAG_ID, TYPE_LIST
12 14 from boards.utils import get_file_mimetype, get_file_hash
13 from django.db import transaction
14 15
15 16 EXCEPTION_NODE = 'Sync node returned an error: {}.'
16 17 EXCEPTION_DOWNLOAD = 'File was not downloaded.'
@@ -30,6 +31,7 TAG_TITLE = 'title'
30 31 TAG_TEXT = 'text'
31 32 TAG_THREAD = 'thread'
32 33 TAG_PUB_TIME = 'pub-time'
34 TAG_UPDATE_TIME = 'update-time'
33 35 TAG_SIGNATURES = 'signatures'
34 36 TAG_SIGNATURE = 'signature'
35 37 TAG_CONTENT = 'content'
@@ -59,6 +61,8 ID_TYPE_URL = 'url'
59 61
60 62 STATUS_SUCCESS = 'success'
61 63
64 CURRENT_MODEL_VERSION = '1.1'
65
62 66
63 67 logger = logging.getLogger('boards.sync')
64 68
@@ -120,6 +124,9 class SyncManager:
120 124 pub_time = et.SubElement(content_tag, TAG_PUB_TIME)
121 125 pub_time.text = str(post.get_pub_time_str())
122 126
127 update_time = et.SubElement(content_tag, TAG_UPDATE_TIME)
128 update_time.text = str(post.last_edit_time)
129
123 130 if post.tripcode:
124 131 tripcode = et.SubElement(content_tag, TAG_TRIPCODE)
125 132 tripcode.text = post.tripcode
@@ -141,8 +148,6 class SyncManager:
141 148 for file in attachments:
142 149 SyncManager._attachment_to_xml(
143 150 attachments_tag, attachment_refs, file)
144 version_tag = et.SubElement(content_tag, TAG_VERSION)
145 version_tag.text = str(post.version)
146 151
147 152 global_id.content = et.tostring(content_tag, ENCODING_UNICODE)
148 153 global_id.save()
@@ -191,8 +196,8 class SyncManager:
191 196 global_id, exists = GlobalId.from_xml_element(tag_id)
192 197 signatures = SyncManager._verify_model(global_id, content_str, tag_model)
193 198
194 version = int(tag_content.find(TAG_VERSION).text)
195 is_old = exists and global_id.post.version < version
199 update_time = tag_content.find(TAG_UPDATE_TIME).text
200 is_old = exists and global_id.post.last_edit_time < parse_datetime(update_time)
196 201 if exists and not is_old:
197 202 logger.debug('Post {} exists and is up to date.'.format(global_id))
198 203 else:
@@ -204,13 +209,13 class SyncManager:
204 209
205 210 title = tag_content.find(TAG_TITLE).text or ''
206 211 text = tag_content.find(TAG_TEXT).text or ''
207 pub_time = tag_content.find(TAG_PUB_TIME).text
208 212 tripcode_tag = tag_content.find(TAG_TRIPCODE)
209 213 if tripcode_tag is not None:
210 214 tripcode = tripcode_tag.text or ''
211 215 else:
212 216 tripcode = ''
213 217
218 pub_time = tag_content.find(TAG_PUB_TIME).text
214 219 thread = tag_content.find(TAG_THREAD)
215 220 tags = []
216 221 if thread:
@@ -257,15 +262,14 class SyncManager:
257 262 Post.objects.update_post(
258 263 post, title=title, text=text, pub_time=pub_time,
259 264 tags=tags, files=files, file_urls=urls,
260 tripcode=tripcode, version=version)
265 tripcode=tripcode, version=version, last_edit_time=update_time)
261 266 logger.debug('Parsed updated post {}'.format(global_id))
262 267 else:
263 268 Post.objects.import_post(
264 269 title=title, text=text, pub_time=pub_time,
265 270 opening_post=opening_post, tags=tags,
266 271 global_id=global_id, files=files,
267 file_urls=urls, tripcode=tripcode,
268 version=version)
272 file_urls=urls, tripcode=tripcode, last_edit_time=update_time)
269 273 logger.debug('Parsed new post {}'.format(global_id))
270 274
271 275 @staticmethod
@@ -285,8 +289,8 class SyncManager:
285 289 tag_model = et.SubElement(models, TAG_MODEL)
286 290 tag_id = et.SubElement(tag_model, TAG_ID)
287 291 post.global_id.to_xml_element(tag_id)
288 tag_version = et.SubElement(tag_model, TAG_VERSION)
289 tag_version.text = str(post.version)
292 update_time = et.SubElement(tag_model, TAG_UPDATE_TIME)
293 update_time.text = str(post.last_edit_time)
290 294
291 295 return et.tostring(response, ENCODING_UNICODE)
292 296
@@ -372,7 +376,7 class SyncManager:
372 376 request.set(ATTR_VERSION, '1.0')
373 377
374 378 model = et.SubElement(request, TAG_MODEL)
375 model.set(ATTR_VERSION, '1.0')
379 model.set(ATTR_VERSION, CURRENT_MODEL_VERSION)
376 380 model.set(ATTR_NAME, 'post')
377 381
378 382 if opening_post:
@@ -186,7 +186,7 class Thread(models.Model):
186 186 """
187 187 Gets replies with only fields that are used for viewing.
188 188 """
189 return self.get_replies().defer('text', 'last_edit_time', 'version')
189 return self.get_replies().defer('text', 'last_edit_time')
190 190
191 191 def get_top_level_replies(self) -> QuerySet:
192 192 return self.get_replies().exclude(refposts__threads__in=[self])
@@ -64,6 +64,7
64 64 <button name="method" value="subscribe" class="not_fav">★ {% trans "Add to favorites" %}</button>
65 65 {% endif %}
66 66 </form>
67 &bull;
67 68 <form action="{% url 'tag' tag.get_name %}" method="post" class="post-button-form">
68 69 {% if is_hidden %}
69 70 <button name="method" value="unhide" class="fav">{% trans "Show" %}</button>
@@ -71,6 +72,7
71 72 <button name="method" value="hide" class="not_fav">{% trans "Hide" %}</button>
72 73 {% endif %}
73 74 </form>
75 &bull;
74 76 <a href="{% url 'tag_gallery' tag.get_name %}">{% trans 'Gallery' %}</a>
75 77 </p>
76 78 {% if tag.get_description %}
@@ -65,22 +65,22 class KeyTest(TestCase):
65 65 '<models>'
66 66 '<model name="post">'
67 67 '<content>'
68 '<id key="%s" local-id="%d" type="%s" />'
68 '<id key="{}" local-id="{}" type="{}" />'
69 69 '<title>test_title</title>'
70 '<text>[post]%s[/post]</text>'
71 '<thread><id key="%s" local-id="%d" type="%s" /></thread>'
72 '<pub-time>%s</pub-time>'
73 '<version>%s</version>'
74 '</content>' % (
70 '<text>[post]{}[/post]</text>'
71 '<thread><id key="{}" local-id="{}" type="{}" /></thread>'
72 '<pub-time>{}</pub-time>'
73 '<update-time>{}</update-time>'
74 '</content>'.format(
75 75 key.public_key,
76 76 reply_post.id,
77 77 key.key_type,
78 str(post.global_id),
78 post.global_id,
79 79 key.public_key,
80 80 post.id,
81 81 key.key_type,
82 str(reply_post.get_pub_time_str()),
83 post.version,
82 reply_post.get_pub_time_str(),
83 reply_post.last_edit_time,
84 84 ) in response,
85 85 'Wrong XML generated for the GET response.')
86 86
@@ -37,13 +37,13 class SyncTest(TestCase):
37 37 '<models>'
38 38 '<model name="post">'
39 39 '<content>'
40 '<id key="%s" local-id="%d" type="%s" />'
41 '<title>%s</title>'
42 '<text>%s</text>'
43 '<tags><tag>%s</tag></tags>'
44 '<pub-time>%s</pub-time>'
45 '<version>%s</version>'
46 '</content>' % (
40 '<id key="{}" local-id="{}" type="{}" />'
41 '<title>{}</title>'
42 '<text>{}</text>'
43 '<tags><tag>{}</tag></tags>'
44 '<pub-time>{}</pub-time>'
45 '<update-time>{}</update-time>'
46 '</content>'.format(
47 47 post.global_id.key,
48 48 post.global_id.local_id,
49 49 post.global_id.key_type,
@@ -51,7 +51,7 class SyncTest(TestCase):
51 51 post.get_sync_text(),
52 52 post.get_thread().get_tags().first().get_name(),
53 53 post.get_pub_time_str(),
54 post.version,
54 post.last_edit_time,
55 55 ) in response,
56 56 'Wrong response generated for the GET request.')
57 57
@@ -86,13 +86,13 class SyncTest(TestCase):
86 86 '<models>'
87 87 '<model name="post">'
88 88 '<content>'
89 '<id key="%s" local-id="%d" type="%s" />'
90 '<title>%s</title>'
91 '<text>%s</text>'
92 '<tags><tag>%s</tag></tags>'
93 '<pub-time>%s</pub-time>'
94 '<version>%s</version>'
95 '</content>' % (
89 '<id key="{}" local-id="{}" type="{}" />'
90 '<title>{}</title>'
91 '<text>{}</text>'
92 '<tags><tag>{}</tag></tags>'
93 '<pub-time>{}</pub-time>'
94 '<update-time>{}</update-time>'
95 '</content>'.format(
96 96 post.global_id.key,
97 97 post.global_id.local_id,
98 98 post.global_id.key_type,
@@ -100,7 +100,6 class SyncTest(TestCase):
100 100 post.get_sync_text(),
101 101 post.get_thread().get_tags().first().get_name(),
102 102 post.get_pub_time_str(),
103 post.version,
104 103 ) in response,
105 104 'Wrong response generated for the GET request.')
106 105
@@ -128,21 +127,21 class SyncTest(TestCase):
128 127 '<models>'
129 128 '<model>'
130 129 '<id key="{}" local-id="{}" type="{}" />'
131 '<version>{}</version>'
130 '<update-time>{}</update-time>'
132 131 '</model>'
133 132 '<model>'
134 133 '<id key="{}" local-id="{}" type="{}" />'
135 '<version>{}</version>'
134 '<update-time>{}</update-time>'
136 135 '</model>'
137 136 '</models>'.format(
138 137 post.global_id.key,
139 138 post.global_id.local_id,
140 139 post.global_id.key_type,
141 post.version,
140 post.last_edit_time,
142 141 post2.global_id.key,
143 142 post2.global_id.local_id,
144 143 post2.global_id.key_type,
145 post2.version,
144 post2.last_edit_time,
146 145 ) in response_all,
147 146 'Wrong response generated for the LIST request for all posts.')
148 147
@@ -173,13 +172,13 class SyncTest(TestCase):
173 172 '<models>'
174 173 '<model>'
175 174 '<id key="{}" local-id="{}" type="{}" />'
176 '<version>{}</version>'
175 '<update-time>{}</update-time>'
177 176 '</model>'
178 177 '</models>'.format(
179 178 post.global_id.key,
180 179 post.global_id.local_id,
181 180 post.global_id.key_type,
182 post.version,
181 post.last_edit_time,
183 182 ) in response_thread,
184 183 'Wrong response generated for the LIST request for posts of '
185 184 'existing thread.')
@@ -240,13 +239,13 class SyncTest(TestCase):
240 239 '<models>'
241 240 '<model>'
242 241 '<id key="{}" local-id="{}" type="{}" />'
243 '<version>{}</version>'
242 '<update-time>{}</update_time>'
244 243 '</model>'
245 244 '</models>'.format(
246 245 post2.global_id.key,
247 246 post2.global_id.local_id,
248 247 post2.global_id.key_type,
249 post2.version,
248 post2.last_edit_time,
250 249 ) in response_thread,
251 250 'Wrong response generated for the LIST request for posts of '
252 251 'existing thread.')
@@ -97,7 +97,7 Sample response:
97 97 <models>
98 98 <model>
99 99 <id key="id1" type="ecdsa" local-id="1">
100 <version>1</version>
100 <update-time>2017-01-01 00:00:00</update-time>
101 101 </model>
102 102 <model>
103 103 <id key="id1" type="ecdsa" local-id="2" />
@@ -1,6 +1,6
1 1 # 0 Title #
2 2
3 "post" model reference
3 "post" model reference of version 1.1
4 4
5 5 # 1 Description #
6 6
@@ -13,7 +13,7
13 13 * title -- text field.
14 14 * text -- text field.
15 15 * pub-time -- timestamp (TBD: Define format).
16 * version -- when post content changes, version should be incremented.
16 * update -- when post content changes, the update time should be incremented.
17 17
18 18 # 2.2 Optional fields #
19 19
General Comments 0
You need to be logged in to leave comments. Login now