Skip to content

Latest commit

 

History

History
 
 

demo_odoo_tutorial

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

odoo 手把手建立第一個 addons

建議觀看影片, 會更清楚:smile:

以下將介紹這個 addons 的結構

說明

介紹 model

首先是 __manifest__.py, 比較重要的是 depends,

{
    'name': "demo odoo tutorial",
    ......

    # any module necessary for this one to work correctly
    'depends': ['base', 'mail'],

    # always loaded
    'data': [
        'security/security.xml',
        'security/ir.model.access.csv',
        'data/data_demo_odoo.xml',
        'views/menu.xml',
        'views/view.xml',
        'reports/report.xml',
        'views/demo_odoo_template.xml',
    ],
    ......
}

在 odoo 的世界中, 一定會看到某個 addons 依賴 xxx addons, 想簡單一點,

你可以把它想成是模組化(方便管理), data 的部份我等等再回來介紹.

models 資料夾, 裡面有 __init__.pymodels.py,

__init__.py 單純就是 import models.py 而已.

models.py 這邊就很重要了

class DemoOdooTutorial(models.Model):
    _name = 'demo.odoo.tutorial'
    _description = 'Demo Odoo Tutorial'
    _inherit = ['mail.thread', 'mail.activity.mixin'] # track_visibility

odoo 中的 model 主要有幾個, 分別是 AbstractModel、Model、TransientModel,

最基本的 BaseModel, 其實 BaseModel = AbstractModel,

https://www.odoo.com/documentation/13.0/reference/orm.html#abstractmodel

@pycompat.implements_to_string
class BaseModel(MetaModel('DummyModel', (object,), {'_register': False})):
    """ Base class for Odoo models.

    Odoo models are created by inheriting:

    *   :class:`Model` for regular database-persisted models

    *   :class:`TransientModel` for temporary data, stored in the database but
        automatically vacuumed every so often

    *   :class:`AbstractModel` for abstract super classes meant to be shared by
        multiple inheriting models

    The system automatically instantiates every model once per database. Those
    instances represent the available models on each database, and depend on
    which modules are installed on that database. The actual class of each
    instance is built from the Python classes that create and inherit from the
    corresponding model.

    Every model instance is a "recordset", i.e., an ordered collection of
    records of the model. Recordsets are returned by methods like
    :meth:`~.browse`, :meth:`~.search`, or field accesses. Records have no
    explicit representation: a record is represented as a recordset of one
    record.

    To create a class that should not be instantiated, the _register class
    attribute may be set to False.
    """
    ....

今天只會先提到 Model, Model 繼承自 AbstractModel

class Model(AbstractModel):
    """ Main super-class for regular database-persisted Odoo models.

    Odoo models are created by inheriting from this class::

        class user(Model):
            ...

    The system will later instantiate the class once per database (on
    which the class' module is installed).
    """
    _auto = True                # automatically create database backend
    _register = False           # not visible in ORM registry, meant to be python-inherited only
    _abstract = False           # not abstract
    _transient = False          # not transient

_auto = True 會自動在 db 中建立 table.

_name 為 model 的名稱, 請注意幾件事情, model 名稱建議都使用單數, 然後不要使用 _ 分隔名稱,

請使用 . 像是範例中的 demo.odoo.tutorial (在 db 中, table 名稱會顯示 demo_odoo_tutorial, 如下圖)

alt tag

_inherit 在 odoo 中不管是 model 還是 view, 甚至是權限, 都會使用繼承 (這邊先知道這樣即可:smile:).

再來說明 field

    ......
    name = fields.Char('Description', required=True)

    # track_visibility='always' 和 track_visibility='onchange'
    is_done_track_onchange = fields.Boolean(
        string='Is Done?', default=False, track_visibility='onchange')
    name_track_always = fields.Char(string="track_name", track_visibility='always')

    start_datetime = fields.Datetime('Start DateTime', default=fields.Datetime.now())
    stop_datetime = fields.Datetime('End Datetime')

    field_onchange_demo = fields.Char('onchange_demo')
    field_onchange_demo_set = fields.Char('onchange_demo_set', readonly=True)

    # float digits
    # field tutorial
    input_number = fields.Float(string='input number', digits=(10,3))
    ......

track_visibility 為追蹤值的改變, 這也是為甚麼要繼承 mail.thread 以及 mail.activity.mixin 的原因,

如果你有修改值, 會紀錄改變(如下圖),

alt tag

start_datetime field 有 default, 設定為當天的時間,

當建立一筆資料時, 會顯示當下的時間,

alt tag

field_onchange_demo_set field 中的 readonly=True,

你可以發現是無法修改的 (可能是根據其他欄位透過 code 改變它的值)

alt tag

input_number Float field 中的 digits 為設定進位以及小數點, 像這邊是算到小數點第3位並使用10進位

alt tag

下一部份的 code

......
field_compute_demo = fields.Integer(compute="_get_field_compute") # readonly

# field_compute_demo = fields.Integer(compute="_get_field_compute",
#                                     inverse="_set_input_number",
#                                     search="_search_upper")

    _sql_constraints = [
        ('name_uniq', 'unique(name)', 'Description must be unique'),
    ]

    @api.constrains('start_datetime', 'stop_datetime')
    def _check_date(self):
        for data in self:
            if data.start_datetime > data.stop_datetime:
                raise ValidationError(
                    "data.stop_datetime  > data.start_datetime"
                )

    @api.depends('input_number')
    def _get_field_compute(self):
        for data in self:
            data.field_compute_demo = data.input_number * 1000

    def _set_input_number(self):
        for data in self:
            data.input_number = data.field_compute_demo / 1000

    def _search_upper(self, operator, value):
        return [('input_number', operator, value)]

    @api.onchange('field_onchange_demo')
    def onchange_demo(self):
        if self.field_onchange_demo:
            self.field_onchange_demo_set = 'set {}'.format(self.field_onchange_demo)
......

field_compute_demo field 為 compute field, compute field 預設為 readonly,

而且這個 field 預設是不會存在 db 中的 (store=False, 也就是每次都是計算出來的),

如果想要將值保存在 db 中, 需再加上 store=True.

如果你設定 store=False (或是沒指定),

當你去搜尋 field_compute_demo 時, 會發現錯誤,

>>> self.env['demo.odoo.tutorial'].search([('field_compute_demo', '>', 1)])
2022-10-08 14:50:42,851 15224 ERROR odoo odoo.osv.expression: Non-stored field demo.odoo.tutorial.field_compute_demo cannot be searched.
demo.odoo.tutorial(1,2)

雖然有撈出資料, 但是是撈出全部的資料(剛好裡面全部的資料有兩筆).

原因很簡單, 因為這個 field 是 compute 出來的, 在 table 中也沒有這個欄位,

所以不能搜尋.

如果你想要搜尋, 一種簡單方法是設定 store=True, 但這種方法不一定是好的:confused:

(因為如果亂設很可能造成效能上的影響).

另一種方法比較麻煩, 透過定義 search 完成,

官方文件可參考 Computed fields

透過 search 去定義邏輯, 根據其他的 filed 欄位(或邏輯)搜尋出想要的結果,

......
field_compute_demo = fields.Integer(compute="_get_field_compute",
                                    inverse="_set_input_number",
                                    search="_search_upper")
......

def _search_upper(self, operator, value):
    return [('input_number', operator, value)]

定義完 search 之後, 就可以正常對 field_compute_demo 搜尋了:smile:

>>> self.env['demo.odoo.tutorial'].search([('field_compute_demo', '=', 2)])
demo.odoo.tutorial(2,)

前面有說到 compute field 預設為 readonly,

如果今天想要讓他可以編輯, 該怎麼做呢:question:

需要定義 inverse,

......
field_compute_demo = fields.Integer(compute="_get_field_compute",
                                    inverse="_set_input_number",
                                    search="_search_upper")
......

def _set_input_number(self):
    for data in self:
        data.input_number = data.field_compute_demo / 1000

定義完之後, 就可以對 field_compute_demo 進行編輯,

任意改 input_numberfield_compute_demo 都可以互相 trigger.

compute_get_field_compute, 透過 @api.depends 裝飾器的幫忙,

這邊會根據 input_number field 的值 * 1000 之後,

將值餵給 field_compute_demo.

alt tag

特別補充說明一下 onchange 也可以 return 一個 dict.

......
    @api.onchange('field_onchange_demo')
    def onchange_demo(self):
        ......

        # warning message
        result = dict()
        result['warning'] = {
            'title': 'HELLO',
            'message': 'I am warning'
        }
        return result

透過上方的寫法, 使用者會跳出提醒視窗 (但不會中斷使用者)

alt tag

_sql_constraints 這個為設定一些限制(直接寫 postgresql),

避免不允許(錯誤)的資料進入 db,

像這邊設定 name field 必須為 unique,

假如你有重複的 name, 系統就會提醒你(如下圖),

alt tag

def _check_date(self) 這段是另一種方式限制, 透過 @api.constrains 裝飾器的幫忙,

這邊限制了 start_datetime 必須大於 stop_datetime, 否則會出現 error,

alt tag

def onchange_demo(self) 這個則是使用了 @api.onchange 裝飾器的幫忙,

主要是根據 field_onchange_demo 的改變, 將值餵給 field_onchange_demo_set,

注意 view 中要有 force_save="1", 否則儲存時會消失.

(原因是因為 field_onchange_demo_set 設定為 readonly 的關係)

alt tag

你可能會發現 @api.depends@api.onchange 幾乎一樣,

其實主要區分兩個比較容易的方法, 就是 @api.depends 可以使用在 related 欄位,

像是之後會介紹的 Many2one Many2many One2many 之類的.

@api.onchange 只能使用在同一個 model 上.

介紹 security, menu, tree, form

接下來來看 security 這個很重要的資料夾, 既然有了 model,

這樣要如何控制誰有權限讀寫修改刪除呢:question:

就是依靠 ir.model.access.csvsecurity.xml 這個檔案:exclamation:

security.xml

  <record id="module_demo_odoo_tutorial" model="ir.module.category">
    <field name="name">Demo odoo tutorial category</field>
  </record>

  <record id="demo_odoo_tutorial_group_user" model="res.groups">
    <field name="name">User</field>
    <field name="category_id"
           ref="module_demo_odoo_tutorial"/>
    <field name="implied_ids"
           eval="[(4, ref('base.group_user'))]"/>
  </record>

  <record id="demo_odoo_tutorial_group_manager" model="res.groups">
    <field name="name">Manager</field>
    <field name="category_id"
           ref="module_demo_odoo_tutorial"/>
    <field name="implied_ids"
           eval="[(4, ref('demo_odoo_tutorial_group_user'))]"/>
    <field name="users"
           eval="[(4, ref('base.user_root')),
                  (4, ref('base.user_admin'))]"/>
  </record>

通常會先建立一個 category, 然後建立兩個 group, 分別是 User 和 Manager,

這邊使用線性的繼承方式, 也就是 Manager 擁有 User 一切的權限.

implied_ids 也就是繼承, 裡面的數字分別代表不同的意思,


(0, _ , {'field': value}) creates a new record and links it to this one.
(1, id, {'field': value}) updates the values on an already linked record.
(2, id, _) removes the link to and deletes the id related record.
(3, id, _) removes the link to, but does not delete, the id related record. This is usually what you will use to delete related records on many-to-many fields.
(4, id, _) links an already existing record.
(5, _, _) removes all the links, without deleting the linked records.
(6, _, [ids]) replaces the list of linked records with the provided list.

_ 也可以改成 0 or False,

尾巴不相關的可以忽略, 像是 (4, id, _) 也可以寫成 (4, id).

ir.model.access.csv 為管理 user 和 manager CRUD 的權限,

id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
access_demo_odoo_user,Demo Odoo Tutorial User Access,model_demo_odoo_tutorial,demo_odoo_tutorial_group_user,1,0,0,0
access_demo_odoo_manager,Demo Odoo Tutorial Manager Access,model_demo_odoo_tutorial,demo_odoo_tutorial_group_manager,1,1,1,1

比較需要注意的地方是 model_id 的部份, 像這邊的 model 為 demo.odoo.tutorial,

但這邊必須填入 model_demo_odoo_tutorial, 規則很簡單, 就是要前面要補上 model,

然後將全部的 . 改成 _ .

group_id 的部份可以空白, 請看下面這個例子,

代表這個 Access Rights 沒特別指定 group (但通常比較少這樣使用)

id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
access_demo_test,Test Access,model_demo_odoo_tutorial,,1,1,1,1

如果你去 odoo 後台的 Access Rights 查詢, 他會顯示黃色的.

alt tag

到 user 中可以切換 group,

alt tag

接著來看 views 資料夾,

先看 menu.xml

    <!-- demo_odoo_tutorial App Menu -->
    <menuitem id="demo_odoo_tutorial_menu"
              name="Demo Odoo Tutorial" />

    <!-- Action to open the demo_odoo_tutorial -->
    <act_window id="action_odoo_tutorial"
      name="Demo Odoo Tutorial Action"
      res_model="demo.odoo.tutorial"
      view_mode="tree,form"/>

    <!-- Menu item to open the demo_odoo_tutorial -->
    <menuitem id="menu_odoo_tutorial"
              name="Demo Odoo Tutorial"
              action="action_odoo_tutorial"
              parent="demo_odoo_tutorial_menu" />

建立一個 menuitem, 然後去定義它的 Action, Action 中比較重要的是 res_modelview_mode,

res_model 就填入對應的 model, view_mode 先簡單填入 tree 和 form,

在 odoo 中有很多 view, 像是 pivot kanban 之類的.

再來看 view.xml,

這邊指定兩個最簡單的,

首先是 tree, 記得將對應的 model 填進去,

......
  <record id="view_tree_demo_odoo_tutorial" model="ir.ui.view">
    <field name="name">Demo Odoo Tutorial List</field>
    <field name="model">demo.odoo.tutorial</field>
    <field name="arch" type="xml">
      <tree>
        <field name="name"/>
        <field name="name_track_always"/>
        <field name="is_done_track_onchange"/>
        <field name="start_datetime"/>
        <field name="stop_datetime"/>
      </tree>
    </field>
  </record>
......

tree 如下

alt tag

接著是 form, 記得將對應的 model 填進去,

(如果都沒寫, 系統會自己產生對應的 form view, 但很醜:sob:)

......
<record id="view_form_demo_odoo_tutorial" model="ir.ui.view">
    <field name="name">Demo Odoo Tutorial Form</field>
    <field name="model">demo.odoo.tutorial</field>
    <field name="arch" type="xml">
      <form string="Demo Odoo Tutorial">
        <sheet>
          <group>
            <field name="name"/>
            <field name="name_track_always"/>
            <field name="is_done_track_onchange"/>
            <field name="start_datetime"/>
            <field name="stop_datetime"/>
            <field name="field_onchange_demo"/>
            <field name="field_onchange_demo_set" force_save="1"/>
            <!-- <field name="input_number" widget="percentage"/> -->
            <field name="input_number"/>
            <field name="field_compute_demo"/>
          </group>
        </sheet>
        <div class="oe_chatter">
          <field name="message_follower_ids" widget="mail_followers"/>
          <field name="activity_ids" widget="mail_activity"/>
          <field name="message_ids" widget="mail_thread"/>
        </div>
      </form>
    </field>
</record>
......

form 如下

alt tag

請注意最後一段的 message_follower_ids activity_ids message_ids,

這並不是我們所建立的 field, 而是繼承 mail.thread mail.activity.mixin 所擁有的,

這段 code 主要是產生這個區塊

alt tag

最後回到 __manifest__.py 中, 記得將對應的路徑填入 data 中,

# always loaded
    'data': [
        'security/security.xml',
        'security/ir.model.access.csv',
        'data/data_demo_odoo.xml',
        'views/menu.xml',
        'views/view.xml',
        'reports/report.xml',
        'views/demo_odoo_template.xml',
    ],

接著來看 data/data_demo_odoo.xml,

  <record id="demo_odoo_1" model="demo.odoo.tutorial">
      <field name="name">demo_odoo_1</field>
      <field name="name_track_always">demo_name_track_always_1</field>
      <field name="is_done_track_onchange">True</field>
  </record>

  <record id="demo_odoo_2" model="demo.odoo.tutorial">
      <field name="name">demo_odoo_2</field>
      <field name="name_track_always">demo_name_track_always_2</field>
      <field name="is_done_track_onchange">True</field>
  </record>

這邊做的事情就是當你安裝了 addons, 它會預設幫你建立一些相關的資料.

注意:exclamation: 它和 demo/demo.xml 資料夾不太一樣, demo 資料夾是當你有勾選

產生 demo 資料時, 你安裝 addons 會自動產生 demo data (如下圖).

alt tag

介紹 report, controller

再來是報表的部份 reports/report.xml,

這邊是定義 report 的 template,

t-as="o" 你可以定義你喜歡的變數

    <template id="report_demo_odoo_tutorial">
        <t t-call="web.html_container">
            <t t-foreach="docs" t-as="o">
                <t t-call="web.external_layout">
                    <div class="page">
                        <h2>Odoo Report</h2>
                        <div>
                            <strong>Name:</strong>
                            <p t-field="o.name"/>
                        </div>
                        <div>
                            <strong>Name_track_always:</strong>
                            <p t-field="o.name_track_always"/>
                        </div>
                    </div>
                </t>
            </t>
        </t>
    </template>

    ......

後面這段則是定義 report 的檔名, report_type, 指定 model

    <report
        id="action_report_demo"
        string="Demo Report"
        model="demo.odoo.tutorial"
        report_type="qweb-pdf"
        name="demo_odoo_tutorial.report_demo_odoo_tutorial"
        file="demo_odoo_tutorial.report_demo_odoo_tutorial"
        print_report_name="'Demo Report - %s' % ((object.name).replace('/', ''))"
    />

記得要將路徑填入 __manifest__.py

'data': [
        ......
        'reports/report.xml',
        ......
    ],

會顯示在這邊

alt tag

報表如下

alt tag

再來是 controllers 這個資料夾,

如果你學過 Django,Flask 你會發現蠻像的:smile:

因為就是定義 route , 然後撈資料, 最後回傳到對應的 view,

(記得要將 controller 填入 __init__.py 中)

controllers/controllers.py

class DemoOdoo(http.Controller):

    @http.route('/demo/odoo', auth='user')
    def list(self, **kwargs):
        obj = http.request.env['demo.odoo.tutorial']
        objs = obj.search([])
        return http.request.render(
            'demo_odoo_tutorial.demo_odoo_template',{'objs': objs})

至於它的 view, 在 views/demo_odoo_template.xml

<template id="demo_odoo_template" name="Demo odoo List">
  <div id="wrap" class="container">
    <h1>Demo Odoo</h1>
      <t t-foreach="objs" t-as="obj">
        <div class="row">
          <span t-field="obj.name" />,
          <span t-field="obj.is_done_track_onchange" />,
          <span t-field="obj.name_track_always" />
        </div>
      </t>
  </div>
</template>

route 我們定義是 @http.route('/demo/odoo', auth='user'),

auth='user' 代表要登入才可以觀看, 所以只要瀏覽 http://0.0.0.0:8069/demo/odoo/

就會看到下圖,

alt tag

Odoo Controller Website 教學

接著介紹在 Controller 中設定 website=True,

首先, 將你的 controllers/controllers.py 加上 website=True

class DemoOdoo(http.Controller):

    @http.route('/demo/odoo', auth='user', website=True)
    def list(self, **kwargs):
        print(http.request.website.id)
        ......

print(http.request.website.id) 這邊稍微注意一下,

一定要設定 website=True, 才會有 website_id.

然後在 views/demo_odoo_template.xml 中呼叫 t-call="website.layout"

<template id="demo_odoo_template" name="Demo odoo List">
  <t t-call="website.layout">
    ......
  </t>
</template>

__manifest__.py 中也請記得加入 website depend,

這樣就會加上 odoo website 的模版了:smile:

alt tag

Odoo Qweb 教學

除了這些, 在 QWeb 中還可以實作出不少變化:smile:

report 和 controller 中的 view 都是 QWeb, 以下使用 report 中的 view 示範,

繼續來看 views/demo_odoo_template.xml

<template id="report_demo_odoo_tutorial">
    <t t-call="web.html_container">
        <t t-foreach="docs" t-as="o">
            <t t-call="web.external_layout">
                <div class="page">
                    ......
                    <div>
                        <strong>start datetime:</strong>
                        <p t-field="o.start_datetime"/>
                    </div>
                    <div>
                        <strong>stop datetime:</strong>
                        <p t-field="o.stop_datetime" t-options='{"format": "Y/MM/dd"}'/>
                    </div>
                    <div>
                        <strong>custom start datetime:</strong>
                        <p t-esc="o.get_custom_portal_date()"/>
                    </div>
                </div>
            </t>
        </t>
    </t>
</template>

可以透過 t-options='{"format": "Y/MM/dd"}' 來改變日期格式.

也可以透過 model 的方式設定新的邏輯 t-esc="o.get_custom_portal_date()"

model 實作的部份請參考 models/models.py

class DemoOdooTutorial(models.Model):
    ......

    start_datetime = fields.Datetime('Start DateTime', default=fields.Datetime.now())
    ......

    def get_custom_portal_date(self):
        str_time = datetime.strftime(self.start_datetime, '%Y/%m/%d')
        return '>{}<'.format(str_time)

效果如下圖

alt tag

說明 odoo manifest 中的 auto_install

特別說明一下 __manifest__.py 裡的 auto_install,

    'installable': True,
    'auto_install': False,
    'application': True,

auto_install

這個值很重要, 如果你不懂, 建議設定 False, 原因是假如你設定為 True,

它會找到你路徑的全部 addons 中的 __manifest__.py 裡找 depends,

你其實可以把他想成是一種反向的依賴, 很容易不小心被它雷到:scream:

舉個例子來看這個問題, 當你安裝 hr_expense addons 時, sale_expense addons 會自動被安裝起來 ❗❗

hr_expense addons 看不到相關 depends,

alt tag

sale_expense addons 可以看到相關 depends

alt tag

也就是當你安裝 hr_expense 時, 因為 sale_expense 裡的 'auto_install': True,

所以自動會把 sale_expense 裝起來.

以下為 sale_expense__manifest__.py

{
    'name': 'Sales Expense',
    ......
    'depends': ['sale_management', 'hr_expense'],
    ......
    'auto_install': True,
}

odoo testing 教學

在 odoo 的世界中, testing 也扮演一個很重要的角色, 今天就來介紹這個 testing:smile:

詳細說明可參考 testing.

這邊只會介紹 python 端的 testing, js 的部份就請自行看文件:smirk:

TransactionCase

每個 function 執行完畢後都會 roll back, 每個 function 都是獨立的不互相影響.

SingleTransactionCase

全部 function 執行完畢後才會 roll back, function 會互相影響.

SavepointCase

使用在比較大型以及複雜的測試, 通常會搭配 setUpClass() 使用, 這邊就不另外介紹,

可自行使用關鍵字查看 source code 如何規劃:smile:

先來看 TransactionCase

請參考 demo_odoo_tutorial/tests/test_demo_odoo_transactioncase.py

from odoo.exceptions import UserError, AccessError, ValidationError
from odoo.tests.common import TransactionCase, tagged

# @tagged('-standard', 'nice')
class TestDemoOdooTransactionCase(TransactionCase):

    def setUp(self, *args, **kwargs):
        """setUp"""
        super(TestDemoOdooTransactionCase, self).setUp(*args, **kwargs)
        print('Run setUp')

    def test_hello_world(self):
        """test_hello_world"""
        self.assertEqual(0, 0, 'test hello world')

    def test_datetime_validation(self):
        """test_datetime_validation"""
        values = {
            'name': 'hello',
            'start_datetime': '2020-02-01',
            'stop_datetime': '2020-01-01',
        }
        with self.assertRaises(ValidationError):
            self.env['demo.odoo.tutorial'].create(values)

    def test_field_compute_demo(self):
        """test_field_compute_demo"""
        values = {
            'name': 'hello',
            'input_number': 2
        }
        data = self.env['demo.odoo.tutorial'].create(values)
        self.assertEqual(data.field_compute_demo, data.input_number * 1000)

注意 __init__.py 需要 import test_demo_odoo_transactioncase,

tests 資料夾底下的 testing 都必須是 test_ 開頭的,

執行方法為加上 --test-enable, 範例如下

python3 odoo-bin -i demo_odoo_tutorial -d odoo -c /home/twtrubiks/work/odoo12/odoo/config/odoo.conf --test-enable

執行時你會看到下方的輸出

alt tag

注意, 這裡有3個 testing test_hello_world test_datetime_validation test_field_compute_demo

setUp 會被執行三次, 因為每執行一個測試 function, setUp 就會被執行一次.

3個 testing function 也都是獨立的, 互相不干擾.

接著來看 SingleTransactionCase

請參考 demo_odoo_tutorial/tests/test_demo_odoo_singletransactioncase

......
class TestDemoOdooSingleTransactionCase(SingleTransactionCase):

    def setUp(self, *args, **kwargs):
        """setUp"""
        super(TestDemoOdooSingleTransactionCase, self).setUp(*args, **kwargs)
        print('Run setUp')
    ......

執行時你會看到下方的輸出

alt tag

這個範例和 test_demo_odoo_transactioncase.py 是一模一樣的,

只是將它改成繼承 SingleTransactionCase.

但你會發現這個會出現錯誤, 原因是因為 model 中有設定 name_uniq

_sql_constraints = [
    ('name_uniq', 'unique(name)', 'Description must be unique'),
]

而我們兩個 testing 的 name 名稱 (create name) 都是一樣的, 也就是

test_datetime_validation test_field_compute_demo, 所以會發生錯誤.

(在 TransactionCase 沒錯誤是因為它和 SingleTransactionCase 的特性不一樣)

除了這些功能之外, 還可以透過 tagged 這個 decorator 來幫助我們完成其他的需求.

如果不了解 decorator, 可參考 What is the python decorator

(記得將 tagged 的註解取消)

from odoo.tests.common import TransactionCase, tagged

@tagged('-standard', 'nice')
class TestDemoOdooTransactionCase(TransactionCase):

......

如果沒有特別設定, odoo defaults 為 standard,

+ - 則代表啟用或不改用(排除), 像上面這個例子,

代表只有在 nice tag 才會生效, 在 standard 中不會生效的,

因為前面有加個 -, 更多詳細文件可參考 invocation.

範例指令,

代表只執行有 nice tag 的測試,

python3 odoo-bin -i demo_odoo_tutorial -d odoo -c /home/twtrubiks/work/odoo12/odoo/config/odoo.conf --test-enable --test-tags nice

代表執行有 nice 以及 standard tag 的測試,

python3 odoo-bin -i demo_odoo_tutorial -d odoo -c /home/twtrubiks/work/odoo12/odoo/config/odoo.conf --test-enable --test-tags 'standard,nice'

使用 SQL VIEW 定義 model

這部份是比較進階的, 如果你是新手, 請跳過這部份 😏

使用時機, 如果你有比較特別的報表, 或是特別的 pivot 使用原生的 ORM 可能比較不好實現,

這時候可以考慮使用原生的 SQL 來完成.

請參考 models.py 資料夾

......

class DemoOdooTutorialStatistics(models.Model):
    _name = 'demo.odoo.tutorial.statistics'
    _description = 'Demo Odoo Tutorial Statistics'
    _auto = False

    create_uid = fields.Many2one('res.users', 'Created by', readonly=True)
    average_input_number = fields.Float(string="Average Input Number", readonly=True)

    @api.model_cr
    def init(self):
        tools.drop_view_if_exists(self.env.cr, self._table)
        query = """
        CREATE OR REPLACE VIEW demo_odoo_tutorial_statistics AS
        (
            SELECT
                min(demo.id) as id,
                create_uid,
                avg(input_number) AS average_input_number
            FROM
                demo_odoo_tutorial AS demo
            GROUP BY demo.create_uid
        );
        """
        self.env.cr.execute(query)

_auto = False 通常都不會去設定他, 也就是預設都是 True, 代表 table 會由 odoo

幫助我們產生, 所以也不需要額外去實作 init.

但這邊設定了 _auto = False 代表我們要自己去管, 而不是由 odoo 協助,

當然, 也需要我們自己去維護, 必須實作 init.

定義 fields 的部份, 就看你需要 demo.odoo.tutorial 中的哪些資料,

把需要的 fields 填上即可, 又或是透過 compute 自己實現邏輯.

( 這邊都設定 readonly=True, 因為 VIEW 本來就應該是唯讀的,

如果你不了解, 可 google TABLE VS VIEW, 他們是不一樣的 )

而在 init 中, 透過 SQL 實作一個 VIEW.

也請記得必須補上, menu.xml view.xml ir.model.access.csv.

這邊使用 pivot 來呈現,

當我們更新或裝上這個 addons 的時候, 可以先透過 pgadmin4 查看 db,

注意 ❗ 我們看的是 VIEW, 不是 TABLE

alt tag

這個 VIEW 的 code 就是在 init 定義的,

alt tag

這邊補充一下為甚麼要使用 min(demo.id) as id,,

原因是 odoo 規定要產生 id, 否則會噴錯, 當然你也可以填 max(demo.id) as id,,

反正一定要給他一個 id 即可.

實際畫面

alt tag

使用 RAW SQL 說明

前面和大家說明過了有時候會使用原生的 SQL 來完成.

這部份將更詳細的說明 RAW SQL 的使用方法以及應該注意的事項:smile:

可參考 models/models.py

......
def demo_raw_sql(self):
    query = """
        SELECT
            id, name,
            is_done_track_onchange,
            name_track_always,
            start_datetime,
            stop_datetime,
            field_onchange_demo,
            field_onchange_demo_set,
            input_number
        FROM
            demo_odoo_tutorial;
    """
    self.env.cr.execute(query)

    print('self.env.cr.fetchall:', self.env.cr.fetchall())
    # print('self.env.cr.fetchone:', self.env.cr.fetchone())
    # print('self.env.cr.dictfetchall:', self.env.cr.dictfetchall())
......

你會發現有三種取值的方法

self.env.cr.fetchall()

alt tag

self.env.cr.fetchone()

alt tag

self.env.cr.dictfetchall()

alt tag

當你在使用 原生的 SQL 時, 要很小心:exclamation::exclamation::exclamation:

因為這種搜尋方式跳過了 ORM 那層, 所以權限以及安全規則的部份都會全被跳過:exclamation:

所以在使用 INSERT/UPDATE 時也不會觸發 create() write(),

所以請特別注意:exclamation::exclamation:

也要小心 SQL注入(SQL injection):exclamation::exclamation:

# SQL injection possible
self.env.cr.execute('SELECT * FROM demo_odoo_tutorial where id >' + '1' + ';')

# good
self.env.cr.execute('SELECT * FROM demo_odoo_tutorial where id > %s;', (1,))

如果今天有使用 like 要注意一下(要加上跳脫字元),

query = """
    SELECT
        id, name,
        is_done_track_onchange,
        name_track_always,
        start_datetime,
        stop_datetime,
        field_onchange_demo,
        field_onchange_demo_set,
        input_number
  FROM
        demo_odoo_tutorial
  WHERE
        name like '%%odoo%%' and 1 = %s;
"""

query_sql_params = (1,)
self.env.cr.execute(query, query_sql_params)

或是使用

query = """
    SELECT
        id, name,
        is_done_track_onchange,
        name_track_always,
        start_datetime,
        stop_datetime,
        field_onchange_demo,
        field_onchange_demo_set,
        input_number
  FROM
        demo_odoo_tutorial
  WHERE
        name like %s and 1 = %s;
"""

query_sql_params = ('%%odoo%%', 1,)
# query_sql_params = ('%odoo%', 1,)

self.env.cr.execute(query, query_sql_params)

更好的方法是使用 更好的處理方式 - SQL string composition