1
|
from datetime import datetime
|
2
|
import logging
|
3
|
from html2text import HTML2Text
|
4
|
from emails.django import Message
|
5
|
from lxml.etree import HTML as HTMLTree
|
6
|
|
7
|
from django.utils import timezone
|
8
|
from django.conf import settings
|
9
|
from django.db import models
|
10
|
from django.core.files.storage import DefaultStorage
|
11
|
from django.utils.translation import ugettext_lazy as _
|
12
|
|
13
|
from ckeditor.fields import RichTextField
|
14
|
|
15
|
channel_choices = (
|
16
|
('mailto', _('Email')),
|
17
|
('homepage', _('Homepage'))
|
18
|
)
|
19
|
|
20
|
logger = logging.getLogger(__name__)
|
21
|
|
22
|
class Category(models.Model):
|
23
|
name = models.CharField(max_length=64, blank=False, null=False)
|
24
|
ctime = models.DateTimeField(auto_now_add=True)
|
25
|
|
26
|
def __unicode__(self):
|
27
|
return self.name
|
28
|
|
29
|
|
30
|
class Announce(models.Model):
|
31
|
category = models.ForeignKey('Category', verbose_name=_('category'))
|
32
|
title = models.CharField(_('title'), max_length=256,
|
33
|
help_text=_('maximum 256 characters'))
|
34
|
text = RichTextField(_('Content'))
|
35
|
publication_time = models.DateTimeField(_('publication time'), blank=True,
|
36
|
null=True)
|
37
|
expiration_time = models.DateTimeField(_('Expires on'), blank=True,
|
38
|
null=True)
|
39
|
ctime = models.DateTimeField(_('creation time'), auto_now_add=True)
|
40
|
mtime = models.DateTimeField(_('modification time'), auto_now=True)
|
41
|
|
42
|
def __unicode__(self):
|
43
|
return u'{title} ({id}) at {mtime}'.format(title=self.title,
|
44
|
id=self.id, mtime=self.mtime)
|
45
|
|
46
|
def is_expired(self):
|
47
|
if self.expiration_time:
|
48
|
return self.expiration_time < timezone.now()
|
49
|
return False
|
50
|
|
51
|
def is_published(self):
|
52
|
if self.is_expired():
|
53
|
return False
|
54
|
if self.publication_time:
|
55
|
return self.publication_time <= timezone.now()
|
56
|
return False
|
57
|
|
58
|
class Meta:
|
59
|
verbose_name = _('announce')
|
60
|
ordering = ('-mtime',)
|
61
|
|
62
|
|
63
|
class Broadcast(models.Model):
|
64
|
announce = models.ForeignKey(Announce, verbose_name=_('announce'))
|
65
|
deliver_time = models.DateTimeField(_('Deliver time'), null=True)
|
66
|
result = models.TextField(_('result'), blank=True)
|
67
|
|
68
|
def __unicode__(self):
|
69
|
if self.deliver_time:
|
70
|
return u'announce {id} delivered via at {time}'.format(
|
71
|
id=self.announce.id, time=self.deliver_time)
|
72
|
return u'announce {id} to deliver'.format(id=self.announce.id)
|
73
|
|
74
|
def send(self):
|
75
|
subscriptions = self.announce.category.subscription_set.all()
|
76
|
total_sent = 0
|
77
|
handler = HTML2Text()
|
78
|
m = Message(html=self.announce.text, subject=self.announce.title,
|
79
|
text=handler.handle(self.announce.text),
|
80
|
mail_from=settings.CORBO_DEFAULT_FROM_EMAIL)
|
81
|
html_tree = HTMLTree(self.announce.text)
|
82
|
storage = DefaultStorage()
|
83
|
for img in html_tree.xpath('//img/@src'):
|
84
|
img_path = img.lstrip(storage.base_url)
|
85
|
m.attach(filename=img, data=storage.open(img_path))
|
86
|
m.attachments[img].is_inline = True
|
87
|
m.transformer.synchronize_inline_images()
|
88
|
m.transformer.load_and_transform()
|
89
|
m.transformer.save()
|
90
|
for s in subscriptions:
|
91
|
if not s.identifier:
|
92
|
continue
|
93
|
sent = m.send(to=s.identifier)
|
94
|
if sent:
|
95
|
total_sent += 1
|
96
|
logger.info('Announce "%s" sent to %s', self.announce.title, s.identifier)
|
97
|
else:
|
98
|
logger.warning('Error occured while sending announce "%s" to %s.',
|
99
|
self.announce.title, s.identifier)
|
100
|
self.result = total_sent
|
101
|
self.deliver_time = timezone.now()
|
102
|
self.save()
|
103
|
|
104
|
class Meta:
|
105
|
verbose_name = _('sent')
|
106
|
ordering = ('-deliver_time',)
|
107
|
|
108
|
|
109
|
class Subscription(models.Model):
|
110
|
category = models.ForeignKey('Category', verbose_name=_('Category'))
|
111
|
uuid = models.CharField(_('User identifier'), max_length=128, blank=True)
|
112
|
identifier = models.CharField(_('identifier'), max_length=128, blank=True,
|
113
|
help_text=_('ex.: mailto, homepage, ...'))
|
114
|
class Meta:
|
115
|
unique_together = ('category', 'identifier', 'uuid')
|