2021-01-20 odoo序列號生成方法:ir.sequence模型

1. 應(yīng)用場景

ERP系統(tǒng)中需要產(chǎn)生序列號的地方很多亥至,最常見的是各種單據(jù)的單號悼沈、商品的編碼、條碼等等姐扮。這些號碼都需要具有唯一性和連續(xù)遞增性絮供,且根據(jù)具體情況,需要有一定的規(guī)則(比如前綴中含有單據(jù)類型碼茶敏、操作員碼壤靶、日期等)。

2. 手工生成序號

單據(jù)創(chuàng)建時惊搏,將會調(diào)用模型父類的create方法贮乳。我們可以在模型中覆寫父類的create方法,并計算當(dāng)前單據(jù)類型的最新序號恬惯。比如下面這段代碼:

    @api.model
    def create(self,values):
        self.env.cr.execute('select * from xxx_model order by order_code desc limit 1')
        x=self.env.cr.dictfetchall()
        if len(x)==0:
            values['order_code']='0000001'
        else:
            xx=int(x[0]['order_code'])
            xx+=1
            xxx=str(xx)
            z=xxx
            for i in range(6-len(xxx)):
                z='0'+z
            values['order_code']=z
        return super().create(values)

self.env.cr.execute()函數(shù)可以直接執(zhí)行sql語句向拆,但這是一種惰性查詢,真正觸發(fā)數(shù)據(jù)庫查詢動作的是x=self.env.cr.dictfetchall()酪耳,這條語句將查詢結(jié)果賦值給變量x亲铡。這條sql語句的意思是:查找xxx_model中order_code最大的一條記錄。如果沒有找到該記錄葡兑,說明當(dāng)前是首張單據(jù)奖蔓,則給其賦值‘0000001’。若找到記錄讹堤,則將該記錄的order_code轉(zhuǎn)換成int類型吆鹤,加1后再按規(guī)則首位補0,補齊7位洲守。
由于odoo ORM實現(xiàn)了serializable級別的事務(wù)隔離疑务,因此這種生成序列號的方式是沒有問題的沾凄,否則需要通過加鎖的方式確保多用戶同時創(chuàng)建單據(jù)時數(shù)據(jù)庫一致性問題。

3. ir.sequence創(chuàng)建序列號

odoo為我們提供了一個很好用的序列號生成工具ir.sequence知允,使用該工具只需要在數(shù)據(jù)文件中注冊一條ir.sequence記錄撒蟀,在模型代碼中就可以調(diào)用該序列。

3.1 注冊ir.sequence記錄

在需要生成序列號的模型視圖文件中温鸽,添加一條ir.sequence記錄保屯,并可定義序列的前綴規(guī)則,如下:

<record id="seqence_b_purchase_order" model='ir.sequence'>
        <field name='name'>Purchase Order</field>
        <field name='code'>sequence.purchase_order</field>
        <field name='prefix'>%(year)s%(month)s%(day)s</field>
        <field name='padding'>5</field>
    </record>

prefix字段定義了前綴涤垫,前綴的寫法是一段格式化字符串姑尺,如下是ir.sequence模型中處理前綴的方法,可以看到蝠猬,%(year)s會被解析為%Y切蟋,并通過date的strftime函數(shù)解析出四位數(shù)的年份,month榆芦、day...的處理方法都是如此柄粹。因此,這里可以寫的參數(shù)包括year匆绣、month镰惦、day、y犬绒、doy(day of year)旺入、woy(week of year)、weekday凯力、h23茵瘾、h12、min咐鹤、sec這10種的組合拗秘。還可以添加一些字符串前綴,比如SN祈惶,用來標(biāo)識單據(jù)類別雕旨。

def _get_prefix_suffix(self, date=None, date_range=None):
        def _interpolate(s, d):
            return (s % d) if s else ''

        def _interpolation_dict():
            now = range_date = effective_date = datetime.now(pytz.timezone(self._context.get('tz') or 'UTC'))
            if date or self._context.get('ir_sequence_date'):
                effective_date = fields.Datetime.from_string(date or self._context.get('ir_sequence_date'))
            if date_range or self._context.get('ir_sequence_date_range'):
                range_date = fields.Datetime.from_string(date_range or self._context.get('ir_sequence_date_range'))

            sequences = {
                'year': '%Y', 'month': '%m', 'day': '%d', 'y': '%y', 'doy': '%j', 'woy': '%W',
                'weekday': '%w', 'h24': '%H', 'h12': '%I', 'min': '%M', 'sec': '%S'
            }
            res = {}
            for key, format in sequences.items():
                res[key] = effective_date.strftime(format)
                res['range_' + key] = range_date.strftime(format)
                res['current_' + key] = now.strftime(format)

            return res

        d = _interpolation_dict()
        try:
            interpolated_prefix = _interpolate(self.prefix, d)
            interpolated_suffix = _interpolate(self.suffix, d)
        except ValueError:
            raise UserError(_('Invalid prefix or suffix for sequence \'%s\'') % (self.get('name')))
        return interpolated_prefix, interpolated_suffix

除了prefix,也可以有suffix(后綴)捧请、padding凡涩、number_next等重要字段。
padding表示序列號數(shù)字部分的位數(shù)疹蛉,比如padding=5活箕,則序列號為[prefix]00001這樣的類型。
number_next表示序列號的遞增數(shù)可款,比如number_next=1育韩,則下一個序列號比上一個序列號增1克蚂。

3.2 模型中使用序列號

注冊了序列號后,就可以在模型中使用筋讨。需要產(chǎn)生序列號的地方一般在模型的create函數(shù)中埃叭。如下這段代碼:

    @api.model
    def create(self,values):
        try:
            values['order_code']=self.env['ir.sequence'].next_by_code('sequence.purchase_order')
            stock=self.env['sunrise.u.stock']
            if 'order_items' in values.keys():
                for item in values['order_items']:
                    stock_commodity=stock.search([('commodity','=',item[2]['commodity'])])
                    stock_commodity.amount+=item[2]['amount']
            return super().create(values)
        except:
            raise Exception('b_purchase_order.py:create()')

try中的第一行則使用了上文注冊的序列號,通過self.env['ir.sequence'].next_by_code('sequence.purchase_order')悉罕,可獲取注冊id="seqence_b_purchase_order"這條ir.sequence對象的next_by_code方法赤屋,獲取到最新序號。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蛮粮,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子谜慌,更是在濱河造成了極大的恐慌然想,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件欣范,死亡現(xiàn)場離奇詭異变泄,居然都是意外死亡,警方通過查閱死者的電腦和手機恼琼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進(jìn)店門妨蛹,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人晴竞,你說我怎么就攤上這事蛙卤。” “怎么了噩死?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵颤难,是天一觀的道長。 經(jīng)常有香客問我已维,道長行嗤,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任垛耳,我火速辦了婚禮栅屏,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘堂鲜。我一直安慰自己栈雳,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布缔莲。 她就那樣靜靜地躺著甫恩,像睡著了一般。 火紅的嫁衣襯著肌膚如雪酌予。 梳的紋絲不亂的頭發(fā)上磺箕,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天奖慌,我揣著相機與錄音,去河邊找鬼松靡。 笑死简僧,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的雕欺。 我是一名探鬼主播岛马,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼屠列!你這毒婦竟也來了啦逆?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤笛洛,失蹤者是張志新(化名)和其女友劉穎夏志,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體苛让,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡沟蔑,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了狱杰。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瘦材。...
    茶點故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖仿畸,靈堂內(nèi)的尸體忽然破棺而出食棕,到底是詐尸還是另有隱情,我是刑警寧澤错沽,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布宣蠕,位于F島的核電站,受9級特大地震影響甥捺,放射性物質(zhì)發(fā)生泄漏抢蚀。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一镰禾、第九天 我趴在偏房一處隱蔽的房頂上張望皿曲。 院中可真熱鬧,春花似錦吴侦、人聲如沸屋休。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽劫樟。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間叠艳,已是汗流浹背奶陈。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留附较,地道東北人吃粒。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像拒课,于是被迫代替她去往敵國和親徐勃。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,713評論 2 354

推薦閱讀更多精彩內(nèi)容