source: main/waeup.ikoba/trunk/src/waeup/ikoba/customers/contracts.py @ 12281

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

Change contract_id generation algorithm. Use Universally Unique IDentifiers instead of consecutive numbers.

  • Property svn:keywords set to Id
File size: 6.0 KB
RevLine 
[12089]1## $Id: contracts.py 12258 2014-12-18 14:44:30Z henrik $
2##
3## Copyright (C) 2014 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##
18"""
[12097]19Customer contract components.
[12089]20"""
21import os
22import grok
23from zope.component import queryUtility, getUtility
24from zope.component.interfaces import IFactory
25from zope.interface import implementedBy
[12144]26from zope.schema import getFields
[12089]27from hurry.workflow.interfaces import IWorkflowInfo, IWorkflowState
28from waeup.ikoba.interfaces import MessageFactory as _
[12258]29from waeup.ikoba.interfaces import (
30    IIkobaUtils, IObjectHistory, VERIFIED, IIDSource)
[12089]31from waeup.ikoba.customers.interfaces import (
[12097]32    IContractsContainer, ICustomerNavigation,
33    IContract, IContractEdit, ICustomersUtils,
[12103]34    ISampleContract, ISampleContractEdit)
[12097]35from waeup.ikoba.customers.utils import generate_contract_id
[12089]36from waeup.ikoba.utils.helpers import attrs_to_fields
37
[12097]38class ContractsContainer(grok.Container):
39    """This is a container for customer contracts.
[12089]40    """
[12097]41    grok.implements(IContractsContainer, ICustomerNavigation)
42    grok.provides(IContractsContainer)
[12089]43
[12097]44    def addContract(self, contract):
45        if not IContract.providedBy(contract):
[12089]46            raise TypeError(
[12097]47                'ContractsContainers contain only IContract instances')
48        self[contract.contract_id] = contract
[12089]49        return
50
51    @property
52    def customer(self):
53        return self.__parent__
54
55    def writeLogMessage(self, view, message):
56        return self.__parent__.writeLogMessage(view, message)
57
[12097]58ContractsContainer = attrs_to_fields(ContractsContainer)
[12089]59
[12097]60class ContractBase(grok.Container):
61    """This is a customer contract baseclass.
[12089]62    """
[12097]63    grok.implements(IContract, IContractEdit, ICustomerNavigation)
64    grok.provides(IContract)
[12089]65    grok.baseclass()
66
[12144]67    check_docs_interface = None
68
[12097]69    contract_category = None
[12092]70
[12103]71    form_fields_interface = None
72
73    edit_form_fields_interface = None
74
[12089]75    def __init__(self):
[12097]76        super(ContractBase, self).__init__()
[12089]77        # The site doesn't exist in unit tests
[12258]78        source = getUtility(IIDSource)
79        self.contract_id = unicode(source.get_hex_uuid())
[12094]80        self.last_product_id = None
[12089]81        return
82
83    @property
84    def history(self):
85        history = IObjectHistory(self)
86        return history
87
88    @property
89    def state(self):
90        return IWorkflowState(self).getState()
91
92    @property
93    def translated_state(self):
94        try:
95            TRANSLATED_STATES = getUtility(
[12097]96                ICustomersUtils).TRANSLATED_CONTRACT_STATES
[12089]97            return TRANSLATED_STATES[self.state]
98        except KeyError:
99            return
100
101    @property
102    def class_name(self):
103        return self.__class__.__name__
104
105    @property
106    def formatted_transition_date(self):
107        try:
[12210]108            return self.history.messages[-1].split(' - ')[0]
109        except IndexError:
[12089]110            return
111
112    @property
113    def customer(self):
114        try:
115            return self.__parent__.__parent__
116        except AttributeError:
117            return None
118
119    def writeLogMessage(self, view, message):
120        return self.__parent__.__parent__.writeLogMessage(view, message)
121
122    @property
[12167]123    def is_editable_by_customer(self):
[12089]124        try:
125            # Customer must be approved
126            cond1 = self.customer.state in getUtility(
[12099]127                ICustomersUtils).CONMANAGE_CUSTOMER_STATES
[12097]128            # Contract must be in state created
[12089]129            cond2 = self.state in getUtility(
[12099]130                ICustomersUtils).CONMANAGE_CONTRACT_STATES
[12089]131            if not (cond1 and cond2):
132                return False
133        except AttributeError:
134            pass
135        return True
136
137    @property
[12144]138    def is_approvable(self):
139        for key, field in getFields(self.check_docs_interface).items():
140            if key.endswith('_object'):
141                obj = getattr(self, key, None)
142                state = getattr(obj, 'state', None)
143                if state and state != VERIFIED:
[12168]144                    return False, _("Attached documents must be verified first.")
145        return True, None
[12144]146
147    @property
[12089]148    def translated_class_name(self):
149        try:
[12099]150            CONTYPES_DICT = getUtility(ICustomersUtils).CONTYPES_DICT
151            return CONTYPES_DICT[self.class_name]
[12089]152        except KeyError:
153            return
154
155
[12097]156class SampleContract(ContractBase):
157    """This is a sample contract.
[12089]158    """
159
[12103]160    grok.implements(ISampleContract, ISampleContractEdit, ICustomerNavigation)
161
[12097]162    contract_category = 'sample'
[12092]163
[12103]164    form_fields_interface = ISampleContract
165
166    edit_form_fields_interface = ISampleContractEdit
167
[12144]168    check_docs_interface = ISampleContract
169
[12097]170SampleContract = attrs_to_fields(SampleContract)
[12089]171
172
[12097]173# Contracts must be importable. So we might need a factory.
174class SampleContractFactory(grok.GlobalUtility):
175    """A factory for contracts.
[12089]176    """
177    grok.implements(IFactory)
[12097]178    grok.name(u'waeup.SampleContract')
179    title = u"Create a new contract.",
180    description = u"This factory instantiates new sample contract instances."
[12089]181
182    def __call__(self, *args, **kw):
[12097]183        return SampleContract(*args, **kw)
[12089]184
185    def getInterfaces(self):
[12097]186        return implementedBy(SampleContract)
[12089]187
[12097]188@grok.subscribe(IContract, grok.IObjectAddedEvent)
189def handle_contract_added(contract, event):
190    """If an contract is added the transition create is fired.
[12090]191    The latter produces a logging message.
192    """
[12097]193    if IWorkflowState(contract).getState() is None:
194        IWorkflowInfo(contract).fireTransition('create')
[12090]195    return
Note: See TracBrowser for help on using the repository browser.