You need an instance of the OERP class to dialog with an OpenERP server. Let’s pretend that you want to connect as admin on the db_name database of the local OpenERP server (with the XML-RPC service which listens on port 8071). First, prepare the connection:
>>> import oerplib
>>> oerp = oerplib.OERP(server='localhost', protocol='xmlrpc', port=8071)
You can also specify the default database to use with the database parameter:
>>> oerp = oerplib.OERP(server='localhost', database='db_name', protocol='xmlrpc', port=8071)
To check databases available, use the oerp.db attribute with the list method:
>>> oerp.db.list()
['db_name', 'db_name2', ...]
The connection is ready, you are able to log in on the server with the account of your choice:
>>> user = oerp.login(user='admin', passwd='admin')
Or, if no default database was specified before:
>>> user = oerp.login(user='admin', passwd='admin', database='db_name')
The login method returns an object representing the user connected. It is built from the server-side model res.users, and all its informations are accessible (see Browse records section):
>>> print(user.name) # print the full name of the user
>>> print(user.company_id.name) # print the name of its company
Now you are connected, you can easily execute any kind of RPC queries on the the OpenERP server (execute model methods, trigger workflow, download reports, and handle wizards).
The basic method to execute queries (related to the /object RPC service) is execute. It takes at least two parameters (model name and the method name) following by variable parameters according to the method called. Example:
>>> order_data = oerp.execute('sale.order', 'read', [1], ['name'])
This instruction will call the read method of the model sale.order with the parameters [1] (list of record IDs) and ['name'] (list of fields to return).
However, for usual methods such as create, read, write, unlink and search there are convenient shortcuts available (see oerplib.OERP):
>>> partner_id = oerp.create('res.partner', {'name': 'Jacky Bob', 'lang': 'fr_FR'})
>>> partner_data = oerp.read('res.partner', [partner_id], ['name'])
>>> oerp.write('res.partner', [partner_id], {'name': 'Charly Bob'})
True
>>> partner_ids = oerp.search('res.partner', [('name', 'ilike', 'Bob')])
>>> oerp.unlink('res.partner', [partner_id])
True
There is another way to perform all methods of a model, with the get method, which provide an API almost syntactically identical to the OpenERP server side API (see oerplib.service.osv.Model):
>>> user_obj = oerp.get('res.users')
>>> user_obj.write([1], {'name': "Dupont D."})
True
>>> context = user_obj.context_get()
>>> context
{'lang': 'fr_FR', 'tz': False}
>>> product_obj = oerp.get('product.product')
>>> product_obj.name_get([3, 4])
[[3, '[PC1] PC Basic'], [4, u'[PC2] Basic+ PC (assembl\xe9 sur commande)']]
If you run an OpenERP version 6.1 or above, the user context is automatically sent. You can disable this behaviour with the oerplib.OERP.config property:
>>> oerp.config['auto_context'] = False
>>> product_obj.name_get([3, 4]) # Without context, lang 'en_US' by default
[[3, '[PC1] Basic PC'], [4, '[PC2] Basic+ PC (assembly on order)']]
Note
The auto_context option only affect model methods.
Here is another example of how to install a module (you have to be logged as an administrator to perform this task). The button_immediate_install method used here is available since OpenERP v6.1:
>>> module_obj = oerp.get('ir.module.module')
>>> module_id = module_obj.search([('name', '=', 'purchase')])
>>> module_obj.button_immediate_install(module_id)
A great functionality of OERPLib is its ability to generate objects that are similar to browsable records found on the OpenERP server. All this is possible using the browse method:
# fetch one record
partner = oerp.browse('res.partner', 1) # Partner ID = 1
print(partner.name)
# fetch several records
for partner in oerp.browse('res.partner', [1, 2]):
print(partner.name)
From such objects, it is possible to easily explore relationships. The related records are generated on the fly:
partner = oerp.browse('res.partner', 3)
for addr in partner.address:
print(addr.name)
You can browse objects through a model too. In fact, both methods are strictly identical, oerplib.OERP.browse() is simply a shortcut to the other:
>>> partner1 = oerp.browse('res.partner', 3)
>>> partner2 = oerp.get('res.partner').browse(3)
>>> partner1 == partner2
True
Outside relation fields, Python data types are used, like datetime.date and datetime.datetime:
>>> order = oerp.browse('purchase.order', 42)
>>> order.minimum_planned_date
datetime.datetime(2012, 3, 10, 0, 0)
>>> order.date_order
datetime.date(2012, 3, 8)
A list of data types used by browse_record fields are available here.
Update data of a browsable record is workable with the write_record method of an OERP instance. Let’s update the first contact’s name of a partner:
>>> addr = list(partner.address)[0] # Get the first address
>>> addr.name = "Caporal Jones"
>>> oerp.write_record(addr)
This is equivalent to:
>>> addr_id = list(partner.address)[0].id
>>> oerp.write('res.partner.address', [addr_id], {'name': "Caporal Jones"})
As see above, it’s as simple as that:
>>> partner.name = "OpenERP"
>>> oerp.write_record(partner)
Same as above, except there is a check about the value assigned. For instance, the field type of the res.partner.address model accept values contains in ['default', 'invoice', 'delivery', 'contact', 'other']:
>>> my_partner_address.type = 'default' # Ok
>>> my_partner_address.type = 'foobar' # Error!
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "oerplib/fields.py", line 58, in setter
value = self.check_value(value)
File "oerplib/fields.py", line 73, in check_value
field_name=self.name,
ValueError: The value 'foobar' supplied doesn't match with the possible values '['default', 'invoice', 'delivery', 'contact', 'other']' for the 'type' field
You can also update a many2one field, with either an ID or a browsable record:
>>> addr.partner_id = 42 # with an ID
>>> oerp.write_record(addr)
>>> partner = oerp.browse('res.partner', 42) # with a browsable record
>>> addr.partner_id = partner
>>> oerp.write_record(addr)
You can’t put any ID or browsable record, a check is made on the relationship to ensure data integrity:
>>> user = oerp.browse('res.users', 1)
>>> addr = oerp.browse('res.partner.address', 1)
>>> addr.partner_id = user
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "oerplib/fields.py", line 128, in setter
o_rel = self.check_value(o_rel)
File "oerplib/fields.py", line 144, in check_value
field_name=self.name))
ValueError: Instance of 'res.users' supplied doesn't match with the relation 'res.partner' of the 'partner_id' field.
one2many and many2many fields can be updated by providing a list of tuple as specified in the OpenERP documentation:
>>> user = oerp.browse('res.users', 1)
>>> user.groups_id = [(6, 0, [8, 5, 6, 4])]
>>> oerp.write_record(user)
To update a reference field, you have to use either a string or a browsable record as below:
>>> helpdesk = oerp.browse('crm.helpdesk', 1)
>>> helpdesk.ref = 'res.partner,1' # with a string with the format '{relation},{id}'
>>> oerp.write_record(helpdesk)
>>> partner = oerp.browse('res.partner', 1)
>>> helpdesk.ref = partner # with a browsable record
>>> oerp.write_record(helpdesk)
A check is made on the relation name:
>>> helpdesk.ref = 'foo.bar,42'
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "oerplib/service/osv/fields.py", line 213, in __set__
value = self.check_value(value)
File "oerplib/service/osv/fields.py", line 244, in check_value
self._check_relation(relation)
File "oerplib/service/osv/fields.py", line 225, in _check_relation
field_name=self.name,
ValueError: The value 'foo.bar' supplied doesn't match with the possible values '['res.partner', 'calendar.event', 'crm.meeting']' for the 'ref' field
date and datetime fields accept either string values or datetime.date/datetime.datetime objects.
With datetime.date and datetime.datetime objects:
>>> order = oerp.browse('purchase.order', 42)
>>> order.date_order = datetime.date(2011, 9, 20)
>>> order.minimum_planned_date = datetime.datetime(2011, 9, 20, 12, 31, 24)
>>> oerp.write_record(order)
With formated strings:
>>> order.date_order = "2011-09-20" # %Y-%m-%d
>>> order.minimum_planned_date = "2011-09-20 12:31:24" # %Y-%m-%d %H:%M:%S
>>> oerp.write_record(order)
As always, a wrong type will raise an exception:
>>> order.date_order = "foobar"
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "oerplib/fields.py", line 187, in setter
value = self.check_value(value)
File "oerplib/fields.py", line 203, in check_value
self.pattern))
ValueError: Value not well formatted, expecting '%Y-%m-%d' format
Another nice functionnality is the reports generation (related to the /report RPC service) with the report method. You have to supply the name of the report, the name of the model and the ID of the record related:
>>> oerp.report('sale.order', 'sale.order', 1)
'/tmp/oerplib_uJ8Iho.pdf'
>>> oerp.report('webkitaccount.invoice', 'account.invoice', 1)
'/tmp/oerplib_r1W9jG.pdf'
The method will return the path to the generated temporary report file.
You can manage OpenERP databases with the oerplib.OERP.db property. It offers you a dynamic access to all methods of the /db RPC service in order to list, create, drop, dump, restore databases and so on.
Note
You have not to be logged in to perform database management tasks. Instead, you have to use the “super admin” password.
Prepare a connection:
>>> import oerplib
>>> oerp = oerplib.OERP(server='localhost')
At this point, you are able to list databases of this server:
>>> oerp.db.list()
[]
Let’s create a new database:
>>> database_id = oerp.db.create('super_admin_passwd', 'test_db', False, 'fr_FR', 'admin')
The creation process may take some time on the OpenERP server, and you have to wait before using the new database. The state of the creation process is returned by the get_progress method:
>>> database_id = oerp.db.create('super_admin_passwd', 'test_db', False, 'fr_FR', 'admin')
>>> while not oerp.db.get_progress('super_admin_passwd', database_id)[0]
... pass
>>> oerp.login('admin', 'admin', 'test_db')
However, OERPLib simplifies this by providing the create_and_wait method:
>>> oerp.db.create_and_wait('super_admin_passwd', 'test_db', False, 'fr_FR', 'admin')
[{'login': u'admin', 'password': u'admin', 'name': u'Administrator'},
{'login': u'demo', 'password': u'demo', 'name': u'Demo User'}]
Some documentation about methods offered by the OpenERP /db RPC service is available here.
By default, the timeout is set to 120 seconds for all RPC requests. If your requests need a higher timeout, you can set it through the oerplib.OERP.config property:
>>> oerp.config['timeout']
120
>>> oerp.config['timeout'] = 300 # Set the timeout to 300 seconds