source: main/waeup.ikoba/trunk/src/waeup/ikoba/utils/utils.py @ 12034

Last change on this file since 12034 was 11983, checked in by Henrik Bettermann, 10 years ago

Add history attribute and test history.

  • Property svn:keywords set to Id
File size: 8.4 KB
RevLine 
[7358]1## $Id: utils.py 11983 2014-11-18 09:17:13Z henrik $
2##
3## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
4## This program is free software; you can redistribute it and/or modify
5## it under the terms of the GNU General Public License as published by
6## the Free Software Foundation; either version 2 of the License, or
7## (at your option) any later version.
8##
9## This program is distributed in the hope that it will be useful,
10## but WITHOUT ANY WARRANTY; without even the implied warranty of
11## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12## GNU General Public License for more details.
13##
14## You should have received a copy of the GNU General Public License
15## along with this program; if not, write to the Free Software
16## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17##
[11949]18"""General helper utilities for Ikoba.
[7358]19"""
20import grok
[11815]21import psutil
[7365]22import string
[8181]23import pytz
[9866]24from copy import deepcopy
[8181]25from random import SystemRandom as r
[7734]26from zope.i18n import translate
[11949]27from waeup.ikoba.interfaces import IIkobaUtils
28from waeup.ikoba.interfaces import MessageFactory as _
29from waeup.ikoba.smtp import send_mail as send_mail_internally
30from waeup.ikoba.utils.helpers import get_sorted_preferred
[7358]31
[11814]32
33def send_mail(from_name, from_addr,
34              rcpt_name, rcpt_addr,
35              subject, body, config):
[11949]36    """Wrapper for the real SMTP functionality in :mod:`waeup.ikoba.smtp`.
[7382]37
[7471]38    Merely here to stay compatible with lots of calls to this place.
[7400]39    """
[7471]40    mail_id = send_mail_internally(
41        from_name, from_addr, rcpt_name, rcpt_addr,
42        subject, body, config)
[7399]43    return True
44
[11814]45
[7874]46#: A list of phone prefixes (order num, country, prefix).
47#: Items with same order num will be sorted alphabetically.
48#: The lower the order num, the higher the precedence.
49INT_PHONE_PREFIXES = [
50    (99, _('Germany'), '49'),
[11814]51    (1, _('Nigeria'), '234'),
[7874]52    (99, _('U.S.'), '1'),
53    ]
54
[11814]55
56def sorted_phone_prefixes(data=INT_PHONE_PREFIXES, request=None):
[7874]57    """Sorted tuples of phone prefixes.
58
59    Ordered as shown above and formatted for use in select boxes.
60
61    If request is given, we'll try to translate all country names in
62    order to sort alphabetically correctly.
63
64    XXX: This is a function (and not a constant) as different
65    languages might give different orders. This is not tested yet.
66
67    XXX: If we really want to use alphabetic ordering here, we might
68    think about caching results of translations.
69    """
70    if request is not None:
71        data = [
72            (x, translate(y, context=request), z)
73            for x, y, z in data]
74    return tuple([
[11814]75        ('%s (+%s)' % (x[1], x[2]), '+%s' % x[2])
[7874]76        for x in sorted(data)
77        ])
78
[11814]79
[11949]80class IkobaUtils(grok.GlobalUtility):
[7678]81    """A collection of parameters and methods subject to customization.
[7829]82
[7358]83    """
[11949]84    grok.implements(IIkobaUtils)
[7678]85    # This the only place where we define the portal language
86    # which is used for the translation of system messages
87    # (e.g. object histories).
[7744]88    PORTAL_LANGUAGE = 'en'
[7358]89
[7701]90    PREFERRED_LANGUAGES_DICT = {
[11814]91        'en': (1, u'English'),
92        'fr': (2, u'Français'),
93        'de': (3, u'Deutsch'),
94        'ha': (4, u'Hausa'),
95        'yo': (5, u'Yoruba'),
96        'ig': (6, u'Igbo'),
[7701]97        }
98
[7874]99    #: A function to return
100    @classmethod
101    def sorted_phone_prefixes(cls, data=INT_PHONE_PREFIXES, request=None):
102        return sorted_phone_prefixes(data, request)
[7871]103
[7841]104    EXAM_SUBJECTS_DICT = {
[7843]105        'math': 'Mathematics',
106        'computer_science': 'Computer Science',
[7841]107        }
[7836]108
[7917]109    #: Exam grades. The tuple is sorted as it should be displayed in
110    #: select boxes.
111    EXAM_GRADES = (
112        ('A', 'Best'),
113        ('B', 'Better'),
114        ('C', 'Good'),
115        )
[7836]116
[11983]117    DOCUMENT_TYPES = {
[11981]118        'generic': 'Generic Document',
[7681]119        }
120
[9405]121    PAYMENT_CATEGORIES = {
[11981]122        'license': 'License Fee',
[9405]123        }
124
[11800]125    #: Set positive number for allowed max, negative for required min
126    #: avail.
127    #:
128    #: Use integer for bytes value, float for percent
129    #: value. `cpu-load`, of course, accepts float values only.
130    #: `swap-mem` = Swap Memory, `virt-mem` = Virtual Memory,
131    #: `phys-mem` = Physical Memory, `cpu-load` = CPU load in percent.
132    SYSTEM_MAX_LOAD = {
133        'swap-mem': None,
134        'virt-mem': None,
135        'phys-mem': None,
136        'cpu-load': 100.0,
137        }
138
[11814]139    def sendContactForm(self, from_name, from_addr, rcpt_name, rcpt_addr,
140                        from_username, usertype, portal, body, subject):
[7358]141        """Send an email with data provided by forms.
142        """
143        config = grok.getSite()['configuration']
[7734]144        text = _(u"""Fullname: ${a}
145User Id: ${b}
146User Type: ${c}
147Portal: ${d}
[7358]148
[7734]149${e}
150""")
[11814]151        text = _(text, mapping={
152            'a': from_name,
153            'b': from_username,
154            'c': usertype,
155            'd': portal,
156            'e': body})
[11949]157        body = translate(text, 'waeup.ikoba',
[7734]158            target_language=self.PORTAL_LANGUAGE)
[8436]159        if not (from_addr and rcpt_addr):
160            return False
[7400]161        return send_mail(
[11814]162            from_name, from_addr, rcpt_name, rcpt_addr,
163            subject, body, config)
[7359]164
[8181]165    @property
166    def tzinfo(self):
167        # For Nigeria: pytz.timezone('Africa/Lagos')
[9543]168        # For Germany: pytz.timezone('Europe/Berlin')
[8181]169        return pytz.utc
170
[11814]171    def fullname(self, firstname, lastname, middlename=None):
[7477]172        """Full name constructor.
173        """
[7359]174        # We do not necessarily have the middlename attribute
175        if middlename:
[8603]176            name = '%s %s %s' % (firstname, middlename, lastname)
[7359]177        else:
[8603]178            name = '%s %s' % (firstname, lastname)
[11814]179        return string.capwords(
180            name.replace('-', ' - ')).replace(' - ', '-')
[7365]181
182    def genPassword(self, length=8, chars=string.letters + string.digits):
[7477]183        """Generate a random password.
184        """
[7365]185        return ''.join([r().choice(chars) for i in range(length)])
186
[8853]187    def sendCredentials(self, user, password=None, url_info=None, msg=None):
[7399]188        """Send credentials as email.
189
[11947]190        Input is the customer for which credentials are sent and the
[7399]191        password.
192
193        Returns True or False to indicate successful operation.
[7365]194        """
[11949]195        subject = 'Your Ikoba credentials'
[7734]196        text = _(u"""Dear ${a},
[7365]197
[7734]198${b}
[11952]199Application and Registration Portal of
[7734]200${c}.
[7365]201
[7734]202Your user name: ${d}
203Your password: ${e}
[8853]204${f}
[7365]205
206Please remember your user name and keep
207your password secret!
208
[7382]209Please also note that passwords are case-sensitive.
210
[7365]211Regards
[7734]212""")
[7399]213        config = grok.getSite()['configuration']
214        from_name = config.name_admin
[7402]215        from_addr = config.email_admin
[7407]216        rcpt_name = user.title
217        rcpt_addr = user.email
[11814]218        text = _(text, mapping={
219            'a': rcpt_name,
220            'b': msg,
221            'c': config.name,
222            'd': user.name,
223            'e': password,
224            'f': url_info})
[7734]225
[11949]226        body = translate(text, 'waeup.ikoba',
[7734]227            target_language=self.PORTAL_LANGUAGE)
[7399]228        return send_mail(
[11814]229            from_name, from_addr, rcpt_name, rcpt_addr,
230            subject, body, config)
[9987]231
232    def getPaymentItem(self, payment):
233        """Return payment item.
234
235        This method can be used to customize the display_item property method.
236        """
237        return payment.p_item
[11815]238
239    def expensive_actions_allowed(self, type=None, request=None):
240        """Tell, whether expensive actions are currently allowed.
241
242        Check system load/health (or other external circumstances) and
243        locally set values to see, whether expensive actions should be
244        allowed (`True`) or better avoided (`False`).
245
246        Use this to allow or forbid exports, report generations, or
247        similar actions.
248        """
249        max_values = self.SYSTEM_MAX_LOAD
[11816]250        for (key, func) in (
251            ('swap-mem', psutil.swap_memory),
[11818]252            ('virt-mem', psutil.virtual_memory),
[11819]253            ('phys-mem', psutil.phymem_usage),
[11816]254            ):
255            max_val = max_values.get(key, None)
256            if max_val is None:
257                continue
258            mem_val = func()
[11815]259            if isinstance(max_val, float):
[11816]260                # percents
[11821]261                if max_val < 0.0:
262                    max_val = 100.0 + max_val
[11816]263                if mem_val.percent > max_val:
[11815]264                    return False
265            else:
[11816]266                # number of bytes
[11821]267                if max_val < 0:
268                    max_val = mem_val.total + max_val
[11816]269                if mem_val.used > max_val:
[11815]270                    return False
271        return True
Note: See TracBrowser for help on using the repository browser.