分类
Odoo python

Odoo timezone时差调整方法

最近在整合VOIP电话(供应商称为云电话)到奇客大师(GeekerMaster)系统,返回的log说是时间戳错误。供应商说每次post的url的时效性是1分钟,检查发现后,我们每次发送Post请求的时候都会重新产生一遍,应该不存在时间过期的问题。

用 _logger.warning记录更多参数,发现在计算post的url时的时间是不对的,比当前电脑时间早了8个小时。而在系统端用date查询时间是正确的。应该是Odoo系统采用UTC时间的问题,Odoo系统前端显示是当前自己时区的时间,后台数据库用的是UTC时间,据说这是解决全球性使用一个系统的时间差问题的最好方法。

但是国内的供应商用的是北京时间UTC+8时区,所以只好着手解决了:

Odoo调整时差方法一

import datetime
from datetime import timedelta

now = (datetime.datetime.now() + timedelta(hours=8)).strftime("%Y%m%d%H%M%S")

直接加8个小时,直观又方便。后面的时间格式根据要求可以更改。

Odoo调整时差方法二

import datetime
import pytz

user_tz = self.env.user.tz
# local_tz = pytz.timezone(user_tz)
now = datetime.datetime.now(pytz.timezone(
            user_tz)).strftime("%Y%m%d%H%M%S")

前提要在个人配置里设置对你自己的时区,否则无法读取。

这里有个小提示: strftime和 strptime是不同的,一个是输出显示时间的样式,后者是输出时间的格式。

ITGeeker技术奇客最近也调试了阿里云的短信,阿里用的时间格式就和时差无关,用的GMT时间。分享一下用python实现阿里短信所需要的时间格式吧:

import time

now = time.strftime("%Y-%m-%dT%H:%M:%SZ", time.gmtime())

UTC确立以后UTC(GMT)=UTC(+0)作为一般提到的世界时,GMT表示零时区的时刻。 … 格林威治时间(GMT)是一个比较模糊的概念。 在1928年,GMT就是世界时。 在UTC被广泛采用后,我们提到的GMT实际上是UTC时间,或者说零时区的标准时。

所以说用UTC是最好的解决方案。
分类
Odoo python

Python实现人民币大写 金额为负时显示错误时的解决方法

GeekerMaster奇客大师用Odoo开发的一些订单和结算报表都包括金额的人民币大写,采用Num2MoneyFormat的方法,不知道当初是谁分享了这段代码,反正它拯救了很多程序猿的时间。

取自于斯,分享与斯。这里ITGeeker技术奇客也分享一下改写的Odoo人民币大写转换Python代码@api 8(gist@github):

使用此段代码创建一个模块,然后在别的模块调用就可以了,调用代码Sample:

    # compute balance RMB
    @api.one
    @api.depends('balance')
    def compute_balance_cn(self):
        if self.balance:
            self.balance_cn = self.env['geeker.rmb.capital'].Num2MoneyFormat(
                self.balance)

    balance_cn = fields.Char(
        'Renminbi (RMB)', compute=compute_balance_cn, readonly=True, store=True, track_visibility='onchange')

问题: 金额为负时显示大写不正确

当然最好的解决方法是自己修改源代码解决这个问题,但稍显复杂,也不知道是否会引起别的问题,毕竟是抄的别人的代码。如果有大神愿意指点就更好了。

那就抄近路变通方式吧,更改前面的调用代码为:

    # compute balance RMB
    @api.one
    @api.depends('balance')
    def compute_balance_cn(self):
        if self.balance >= 0.00:
            self.balance_cn = self.env['geeker.rmb.capital'].Num2MoneyFormat(
                self.balance)
        else:
            balance_cn_minus = 0 - self.balance
            self.balance_cn = '- ' + self.env['geeker.rmb.capital'].Num2MoneyFormat(
                balance_cn_minus)

    balance_cn = fields.Char(
        'Renminbi (RMB)', compute=compute_balance_cn, readonly=True, track_visibility='onchange')

把负的金额变成正的,然后再计算出来的人民币大写前面加上一个负号“-”,你也可以改成你想要的其他字符,例如“倒欠”等。

分类
Odoo python

Odoo如何获取上个月的第1天和最后一天

GeekerMaster奇客大师有一个SAP自由顾问管理模块,每个月初可以添加上个月的人天数记录,为了减少录入工作量,奇客大师设置了自动获取上个月的工作周期,从月初到月末,代码如下:

import datetime

date_from = fields.Date('Date From', required=True, default=datetime.date(datetime.date.today().year, datetime.date.today().month-1, 1))
date_to = fields.Date('Date To', required=True, default=datetime.date(datetime.date.today().year, datetime.date.today().month, 1)-datetime.timedelta(1))

一直工作正常,结果刚刚进入2019新年,爆出 错误代码:

date_from = fields.Date('Date From', required=True, default=datetime.date(datetime.date.today().year,datetime.date.today().month-1,1))
ValueError: month must be in 1..12

应该是因为现在是1月,如果month-1之后就是0月,系统认为只有1-12个月才存在的缘故吧。使用odoo api进行改造吧:

    @api.multi
    def default_last_month_first_day(self):
        tday = datetime.date.today()
        now_month = tday.month
        if now_month > 1:
            last_month = datetime.date(
                    tday.year, tday.month-1, 1)
        else:
            last_month = datetime.date(
                tday.year, tday.month, 1
            )
        return(last_month)

可以工作,但不完美,也就是如果是1月份,那么date_from获取的是当前月份的第一天,不是上个月的第一天。

引入relativedelta完美解决

import datetime
from dateutil import relativedelta

    @api.multi
    def default_last_month_first_day(self):
        today = datetime.date.today()
        d = today - relativedelta.relativedelta(months=1)
        return datetime.date(d.year, d.month, 1)

    date_from = fields.Date('Date From', required=True, default=default_last_month_first_day, track_visibility='onchange')