# -*- coding: utf-8 # $Id$ # Copyright 2001, 2002 by IVA Team and contributors # # This file is part of IVA. # # IVA is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 of the License, or # (at your option) any later version. # # IVA is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with IVA; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA __version__ = '$Revision$'[11:-2] from types import StringType import time try: from PIL import Image PIL_imported = 1 except ImportError: PIL_imported = 0 import cStringIO import OFS import Globals from Globals import Persistent from AccessControl import ClassSecurityInfo from Thread import EventManager from TraversableWrapper import Traversable from common import intersect_bool, get_roles from common import perm_view, perm_edit, perm_manage, perm_add_lo, translate from common import roles_admin from common import roles_student, roles_tutor, roles_teacher from input_checks import render, normal_entry_tags from zope.component import getUtility from zope.interface import implements from interfaces import IStatistics, IJamArtefact class JamArtefact( Traversable, Persistent, EventManager, OFS.Folder.Folder, OFS.SimpleItem.Item ): """JamArtefact""" meta_type = 'JamArtefact' security= ClassSecurityInfo() security.declareObjectPublic() implements(IJamArtefact) def __init__(self, id_, parent_ids, name, data, content_type): self.id = id_ self.title = name EventManager.__init__(self) self.__name = name self.__data = data self.__censored = 0 self.__content_type = content_type # generate suitable thumbnail/icon depending on content_type subtype = content_type[content_type.find('/')+1:] content_type = content_type[:content_type.find('/')] if content_type == 'image': if not PIL_imported: print "Can't create thumbnail image. PIL not installed." try: s = cStringIO.StringIO(data) im = Image.open(s) im.thumbnail((60, 60)) if im.mode != 'RGB': im = im.convert('RGB') s = cStringIO.StringIO() im.save(s, "JPEG") s.seek(0) self.__thumbnail = s.read() except: self.__icon = 'images/jam_type_img.gif' elif content_type == 'audio': self.__icon = 'images/jam_type_sound.gif' elif content_type == 'text': if subtype == 'html': self.__icon = 'images/jam_type_html.gif' else: self.__icon = 'images/jam_type_text.gif' elif content_type == 'video': self.__icon = 'images/jam_type_video.gif' elif content_type == 'application': # Ideally, we should have all archive format MIME types here.. if subtype in ('gnutar', 'zip', 'x-compress', 'x-compressed', 'x-gtar', 'x-tar', 'x-zip', 'x-zip-compressed', 'x-shar', 'x-bzip', 'x-bzip2', 'x-cpio'): self.__icon = 'images/jam_type_arch.gif' else: self.__icon = 'images/jam_type_no.gif' else: self.__icon = 'images/jam_type_no.gif' self.__annotations = [] if type(parent_ids) == StringType: parent_ids = (parent_ids,) self.__parent_ids = parent_ids security.declarePrivate('manage_afterAdd') def manage_afterAdd(self, item, container): """Set default permissions for roles.""" self.manage_permission(perm_manage, roles_admin, 0) self.manage_permission(perm_edit, roles_teacher, 0) self.manage_permission(perm_add_lo, roles_student, 0) self.manage_permission(perm_view, roles_student, 0) security.declareProtected(perm_view, 'get_children_artefacts') def get_children_artefacts(self): """Return a list of children artefacts.""" children = [] for ja in self.parent().get_children('JamArtefact'): if self.get_id() in ja.get_parent_ids(): children.append(ja) return children security.declareProtected(perm_view, 'get_name') def get_name(self, REQUEST=None): """Return name of the artefact.""" prefix = '' if self.__censored: if REQUEST: uname = str(REQUEST.AUTHENTICATED_USER) prefix = translate(self,'Censored') else: prefix = 'CENSORED' uname='' # If the person is not the author and is not a teacher, # we only show the prefix (the censored notice) if uname!=self.get_author() and \ not self.may_censor_jam_artefact(uname): return prefix prefix = prefix + ' / ' return prefix + self.__name # Return real name of the artefact, even when the artefact is censored. security.declarePrivate('get_real_name') def get_real_name(self): """Return name of the artefact.""" return self.__name security.declareProtected(perm_view, 'get_data') def get_data(self, REQUEST=None, RESPONSE=None): """Return actual data.""" if self.__censored: if not REQUEST or \ not self.may_censor_jam_artefact( str(REQUEST.AUTHENTICATED_USER)): return None if REQUEST: self.update_reader(str(REQUEST.AUTHENTICATED_USER)) if REQUEST and RESPONSE: RESPONSE.setHeader('Content-Type', self.__content_type) return self.__data security.declarePrivate('get_real_data') def get_real_data(self): """Return actual data.""" return self.__data security.declareProtected(perm_view, 'get_content_type') def get_content_type(self): """Return content type.""" return self.__content_type security.declareProtected(perm_view, 'get_icon') def get_icon(self, REQUEST, RESPONSE): """Return icon.""" if self.__censored: return self.unrestrictedTraverse('jam_type_censored.gif').GET() elif hasattr(self, '_' + self.__class__.__name__+ '__icon'): if not self.__icon.endswith('.gif'): self.__icon += '.gif' try: img = self.unrestrictedTraverse(self.__icon) except KeyError: img = self.unrestrictedTraverse('++resource++'+self.__icon) return img.GET() else: RESPONSE.setHeader('Content-Type', 'image/jpeg') return self.__thumbnail security.declareProtected(perm_view, 'get_author') def get_author(self): """Return author (owner) of this item.""" return self.getOwner() security.declareProtected(perm_view, 'get_n_annotations') def get_n_annotations(self): """Return the number of annotations.""" return len(self.__annotations) security.declareProtected(perm_view, 'get_n_censored_annotations') def get_n_censored_annotations(self): """Return the number of censored annotations.""" n = 0 for t in self.__annotations: if t[3]: n += 1 return n security.declareProtected(perm_view, 'get_n_uncensored_annotations') def get_n_uncensored_annotations(self): """Return the number of uncensored annotations.""" n = 0 for t in self.__annotations: if not t[3]: n += 1 return n security.declareProtected(perm_view, 'get_annotations') def get_annotations(self, REQUEST): """Return list of annotation tuples.""" if self.__censored and \ not self.may_censor_jam_artefact(str(REQUEST.AUTHENTICATED_USER)): return [] # FIXME: This have to be changed when we have censoring # FIXME: on the annotation level. anns = [] for (who, when, what, censored, censorer) in self.__annotations: if censored: stamp = time.strftime(translate(self,'short_date_format',default="%Y-%m-%d"), time.localtime(censored)) prefix = 'censored %s %s' % (censorer, stamp) uname = str(REQUEST.AUTHENTICATED_USER) if uname != self.get_author() and \ not self.may_censor_jam_artefact(uname): what = prefix else: what = prefix + \ '
' + render(what, legal_tags=normal_entry_tags, do_horizontal_space=1) else: what = render(what, legal_tags=normal_entry_tags, do_horizontal_space=1) anns.append((who, when, what, censored, censorer)) return anns security.declarePrivate('get_real_annotations') def get_real_annotations(self): """Return list of annotation tuples.""" return self.__annotations security.declarePrivate('add_annotation') def add_annotation(self, uname, annotation_time, annotation_text): """Add annotation to JamArtefact.""" # annotation censorship censored = 0 censorer = '' # uname and annotation_time can be used as a key attribute self.__annotations.append((uname, annotation_time, annotation_text, censored, censorer)) self._p_changed = 1 security.declarePrivate('do_censor_annotations') def do_censor_annotations(self, annotation_indexes, time, user): """Censor annotation.""" for i in [int(x) for x in annotation_indexes]: t = self.__annotations[i] self.__annotations[i] = (t[0], t[1], t[2], time, user) self._p_changed = 1 security.declarePrivate('do_uncensor_annotations') def do_uncensor_annotations(self, annotation_indexes): """Uncensor annotation.""" for i in [int(x) for x in annotation_indexes]: t = self.__annotations[i] self.__annotations[i] = (t[0], t[1], t[2], 0, '') self._p_changed = 1 security.declareProtected(perm_add_lo, 'add_annotation_form_handler') def add_annotation_form_handler( self, REQUEST, annotation_text, submit = '', # form buttons cancel = '', # ): """Add annotation form handler.""" user = str(REQUEST.AUTHENTICATED_USER) if self.__censored and not self.may_censor_jam_artefact(user): raise 'FLE Error', "I don't want to anything." if submit: time_now = time.time() # FIXME: report that annotation_text doesn't work if annotation_text: self.add_annotation(user, time_now, annotation_text) s = getUtility(IStatistics) s._updateStat(REQUEST, 'postedArtefactAnnos') s._updateStat(REQUEST, 'modified_jamming') elif cancel: pass else: raise 'FLE Error', 'Unknown button.' # Should never happen. REQUEST.RESPONSE.redirect(self.absolute_url()) security.declareProtected(perm_view, 'get_parent_ids') def get_parent_ids(self): """Return a list of parent ids.""" if type(self.__parent_ids) == StringType: return (self.__parent_ids,) else: return self.__parent_ids security.declarePrivate('set_parent_ids') def set_parent_ids(self, parent_ids): """Set parent ids.""" self.__parent_ids = parent_ids security.declareProtected(perm_view, 'may_censor_jam_artefact') def may_censor_jam_artefact(self, person): """Return boolean depending on whether user may or may not censor jam artefacts.""" from AccessControl.PermissionRole import rolesForPermissionOn return intersect_bool( get_roles(self, person), rolesForPermissionOn(perm_edit, self)) security.declareProtected(perm_edit, 'censor_jam_artefact_handler') def censor_jam_artefact_handler( self, REQUEST, censor=None, # buttons in jam artefact page uncensor=None, # censor_annotations=None, # uncensor_annotations=None, # verify=None, # buttons in verify page verify_annotations=None, # cancel=None, # annotation_indexes=(), ): """Handles jam_artefact censorship.""" if type(annotation_indexes) == StringType: annotation_indexes = (annotation_indexes,) if verify: import time self.do_censor(time.time(),str(REQUEST.AUTHENTICATED_USER)) if REQUEST: return self.restrictedTraverse('message_dialog.html')( self, REQUEST, title='Message is censored', message='Message is censored', action=self.absolute_url()) if verify_annotations: import time self.do_censor_annotations(annotation_indexes, time.time(), str(REQUEST.AUTHENTICATED_USER)) if REQUEST: return self.restrictedTraverse('message_dialog.html')( self, REQUEST, title='Annotation is censored', message='Annotation is censored', action=self.absolute_url()) elif cancel: if REQUEST: REQUEST.RESPONSE.redirect(self.absolute_url()) elif uncensor: self.do_uncensor() if REQUEST: REQUEST.RESPONSE.redirect(self.absolute_url()) elif uncensor_annotations: self.do_uncensor_annotations(annotation_indexes) if REQUEST: REQUEST.RESPONSE.redirect(self.absolute_url()) elif censor: if REQUEST: return self.restrictedTraverse('message_dialog2.html')( self, REQUEST, title='Warning', message='Are you sure you want to censor this artefact? Your name and this date will be stored and displayed along with the censorship notice to everyone viewing this artefact.', handler='censor_jam_artefact_handler', extra_value_name='jam_artefact', extra_values=(str(self.get_id()),), option1_value = 'cancel', option1_name = 'cancel', option2_value = 'ok', option2_name = 'verify') elif censor_annotations: if REQUEST: return self.restrictedTraverse('message_dialog2.html')( self, REQUEST, title="Warning", message="Are you sure you want to censor the annotations you selected? Your name and this date will be stored and displayed along with the censorship notice to everyone viewing annotations.", handler='censor_jam_artefact_handler', extra_value_name='annotation_indexes', extra_values=annotation_indexes, option1_value = 'Cancel', option1_name = 'cancel', option2_value = 'Ok', option2_name = 'verify_annotations') else: raise 'FLE Error', 'Unknown button' def do_censor(self, time, user): self.__censored = time self.__censorer = user def do_uncensor(self): self.__censored = 0 del self.__censorer security.declareProtected(perm_view, 'is_censored') def is_censored(self): """Return whether this jam artefact is censored.""" return not not self.__censored security.declareProtected(perm_view, 'get_censor_time') def get_censor_time(self, REQUEST): """Return time when the artefact was censored.""" return time.strftime(translate(self,'short_date_format',default="%Y-%m-%d"), time.localtime(self.__censored)) security.declareProtected(perm_view, 'get_censorer') def get_censorer(self): """Return person who censored the artefact.""" return self.__censorer Globals.default__class_init__(JamArtefact) # EOF