1 # -*- encoding: utf-8 -*-
2 ##############################################################################
4 # Copyright (c) 2006 TINY SPRL. (http://tiny.be) All Rights Reserved.
5 # Fabien Pinckaers <fp@tiny.Be>
7 # WARNING: This program as such is intended to be used by professional
8 # programmers who take the whole responsability of assessing all potential
9 # consequences resulting from its eventual inadequacies and bugs
10 # End users who are looking for a ready-to-use solution with commercial
11 # garantees and support are strongly adviced to contract a Free Software
14 # This program is Free Software; you can redistribute it and/or
15 # modify it under the terms of the GNU General Public License
16 # as published by the Free Software Foundation; either version 2
17 # of the License, or (at your option) any later version.
19 # This program is distributed in the hope that it will be useful,
20 # but WITHOUT ANY WARRANTY; without even the implied warranty of
21 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22 # GNU General Public License for more details.
24 # You should have received a copy of the GNU General Public License
25 # along with this program; if not, write to the Free Software
26 # Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
28 ##############################################################################
30 from xml.dom import minidom
31 from osv import fields,osv
36 class base_module_record(osv.osv):
37 _name = "ir.module.record"
41 def __init__(self, pool, cr=None):
42 if super(base_module_record, self).__init__.func_code.co_argcount ==3:
43 super(base_module_record, self).__init__(pool,cr)
45 super(base_module_record, self).__init__(pool)
47 self.recording_data = []
51 def _create_id(self, cr, uid, model, data):
55 name = filter(lambda x: x in string.letters, (data.get('name','') or '').lower())
58 val = model.replace('.','_')+'_'+name+ str(i)
60 if val not in self.ids.values():
64 def _get_id(self, cr, uid, model, id):
65 if (model,id) in self.ids:
66 return self.ids[(model,id)], False
67 dt = self.pool.get('ir.model.data')
68 dtids = dt.search(cr, uid, [('model','=',model), ('res_id','=',id)])
71 obj = dt.browse(cr, uid, dtids[0])
72 self.depends[obj.module] = True
73 return obj.module+'.'+obj.name, obj.noupdate
75 def _create_record(self, cr, uid, doc, model, data, record_id, noupdate=False):
76 record = doc.createElement('record')
77 record.setAttribute("id", record_id)
78 record.setAttribute("model", model)
79 lids = self.pool.get('ir.model.data').search(cr, uid, [('model','=',model)])
80 res = self.pool.get('ir.model.data').read(cr, uid, lids[:1], ['module'])
82 self.depends[res[0]['module']]=True
83 record_list = [record]
84 fields = self.pool.get(model).fields_get(cr, uid)
85 for key,val in data.items():
86 if not (val or (fields[key]['type']=='boolean')):
88 if fields[key]['type'] in ('integer','float'):
89 field = doc.createElement('field')
90 field.setAttribute("name", key)
91 field.setAttribute("eval", val and str(val) or 'False' )
92 record.appendChild(field)
93 elif fields[key]['type'] in ('boolean',):
94 field = doc.createElement('field')
95 field.setAttribute("name", key)
96 field.setAttribute("eval", val and '1' or '0' )
97 record.appendChild(field)
98 elif fields[key]['type'] in ('many2one',):
99 field = doc.createElement('field')
100 field.setAttribute("name", key)
101 if type(val) in (type(''),type(u'')):
104 id,update = self._get_id(cr, uid, fields[key]['relation'], val)
105 noupdate = noupdate or update
107 field.setAttribute("model", fields[key]['relation'])
108 name = self.pool.get(fields[key]['relation']).browse(cr, uid, val).name
109 if isinstance(name, basestring):
110 name = name.decode('utf8')
111 field.setAttribute("search", "[('name','=','"+name+"')]")
113 field.setAttribute("ref", id)
114 record.appendChild(field)
115 elif fields[key]['type'] in ('one2many',):
116 for valitem in (val or []):
118 if key in self.pool.get(model)._columns:
119 fname = self.pool.get(model)._columns[key]._fields_id
121 fname = self.pool.get(model)._inherit_fields[key][2]._fields_id
122 valitem[2][fname] = record_id
123 newid = self._create_id(cr, uid, fields[key]['relation'], valitem[2])
124 childrecord, update = self._create_record(cr, uid, doc, fields[key]['relation'],valitem[2], newid)
125 noupdate = noupdate or update
126 record_list += childrecord
127 self.ids[(fields[key]['relation'],newid)] = newid
130 elif fields[key]['type'] in ('many2many',):
132 for valitem in (val or []):
134 for id2 in valitem[2]:
135 id,update = self._get_id(cr, uid, fields[key]['relation'], id2)
136 self.ids[(fields[key]['relation'],id)] = id
137 noupdate = noupdate or update
139 field = doc.createElement('field')
140 field.setAttribute("name", key)
141 field.setAttribute("eval", "[(6,0,["+','.join(map(lambda x: "ref('%s')" % (x,), res))+'])]')
142 record.appendChild(field)
144 field = doc.createElement('field')
145 field.setAttribute("name", key)
147 if not isinstance(val, basestring):
150 val = val and ('"""%s"""' % val.replace('\\', '\\\\').replace('"', '\"')) or 'False'
151 if isinstance(val, basestring):
152 val = val.decode('utf8')
154 field.setAttribute(u"eval", val)
155 record.appendChild(field)
156 return record_list, noupdate
158 def _generate_object_xml(self, cr, uid, rec, recv, doc, result=None):
163 id,update = self._get_id(cr, uid, rec[3], id)
164 noupdate = noupdate or update
167 record,update = self._create_record(cr, uid, doc, rec[3], rec[6], id)
168 noupdate = noupdate or update
169 record_list += record
170 elif rec[4]=='create':
171 id = self._create_id(cr, uid, rec[3],rec[5])
172 record,noupdate = self._create_record(cr, uid, doc, rec[3], rec[5], id)
173 self.ids[(rec[3],result)] = id
174 record_list += record
175 return record_list,noupdate
176 def _generate_assert_xml(self, rec, doc):
178 def generate_xml(self, cr, uid):
179 # Create the minidom document
181 doc = minidom.Document()
182 terp = doc.createElement("terp")
183 doc.appendChild(terp)
184 for rec in self.recording_data:
185 if rec[0]=='workflow':
186 rec_id,noupdate = self._get_id(cr, uid, rec[1][3], rec[1][5])
189 data = doc.createElement("data")
190 terp.appendChild(data)
191 wkf = doc.createElement('workflow')
192 data.appendChild(wkf)
193 wkf.setAttribute("model", rec[1][3])
194 wkf.setAttribute("action", rec[1][4])
196 data.setAttribute("noupdate", "1")
197 wkf.setAttribute("ref", rec_id)
199 res_list,noupdate = self._generate_object_xml(cr, uid, rec[1], rec[2], doc, rec[3])
200 data = doc.createElement("data")
202 data.setAttribute("noupdate", "1")
204 terp.appendChild(data)
206 data.appendChild(res)
207 elif rec[0]=='assert':
209 res = doc.toprettyxml(indent="\t")
210 return doc.toprettyxml(indent="\t").encode('utf8')
214 def execute(*args, **argv):
215 if len(args) >= 6 and isinstance(args[5], dict):
216 _old_args = args[5].copy()
219 res = fnct(*args, **argv)
220 pool = pooler.get_pool(args[0])
221 mod = pool.get('ir.module.record')
222 if mod and mod.recording:
223 if args[4] not in ('default_get','read','fields_view_get','fields_get','search','search_count','name_search','name_get','get','request_get', 'get_sc'):
224 if _old_args is not None:
225 args[5].update(_old_args)
226 mod.recording_data.append(('query', args, argv,res))
230 def fnct_call_workflow(fnct):
231 def exec_workflow(*args, **argv):
232 res = fnct(*args, **argv)
233 pool = pooler.get_pool(args[0])
234 mod = pool.get('ir.module.record')
235 if mod and mod.recording:
236 mod.recording_data.append(('workflow', args, argv))
240 obj = netsvc._service['object']
241 obj.execute = fnct_call(obj.execute)
242 obj.exportMethod(obj.execute)
243 obj.exec_workflow = fnct_call_workflow(obj.exec_workflow)
244 obj.exportMethod(obj.exec_workflow)
246 # vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4: