runbot/runbot/models/version.py
Xavier ALT fffc27d2fa [FIX] runbot: fix creation of new runbot.version from the backend
Traceback (most recent call last):
  File "/home/odoo/src/odoo/15.0/odoo/addons/base/models/ir_http.py", line 237, in _dispatch
    result = request.dispatch()
  File "/home/odoo/src/odoo/15.0/odoo/http.py", line 687, in dispatch
    result = self._call_function(**self.params)
  File "/home/odoo/src/odoo/15.0/odoo/http.py", line 359, in _call_function
    return checked_call(self.db, *args, **kwargs)
  File "/home/odoo/src/odoo/15.0/odoo/service/model.py", line 94, in wrapper
    return f(dbname, *args, **kwargs)
  File "/home/odoo/src/odoo/15.0/odoo/http.py", line 348, in checked_call
    result = self.endpoint(*a, **kw)
  File "/home/odoo/src/odoo/15.0/odoo/http.py", line 916, in __call__
    return self.method(*args, **kw)
  File "/home/odoo/src/odoo/15.0/odoo/http.py", line 535, in response_wrap
    response = f(*args, **kw)
  File "/home/odoo/src/odoo/15.0/addons/web/controllers/main.py", line 1347, in call_kw
    return self._call_kw(model, method, args, kwargs)
  File "/home/odoo/src/odoo/15.0/addons/web/controllers/main.py", line 1339, in _call_kw
    return call_kw(request.env[model], method, args, kwargs)
  File "/home/odoo/src/odoo/15.0/odoo/api.py", line 464, in call_kw
    result = _call_kw_multi(method, model, args, kwargs)
  File "/home/odoo/src/odoo/15.0/odoo/api.py", line 451, in _call_kw_multi
    result = method(recs, *args, **kwargs)
  File "/home/odoo/src/odoo/15.0/odoo/models.py", line 6489, in onchange
    snapshot1 = Snapshot(record, nametree)
  File "/home/odoo/src/odoo/15.0/odoo/models.py", line 6271, in __init__
    self.fetch(name)
  File "/home/odoo/src/odoo/15.0/odoo/models.py", line 6281, in fetch
    self[name] = record[name]
  File "/home/odoo/src/odoo/15.0/odoo/models.py", line 5888, in __getitem__
    return self._fields[key].__get__(self, type(self))
  File "/home/odoo/src/odoo/15.0/odoo/fields.py", line 1054, in __get__
    self.recompute(record)
  File "/home/odoo/src/odoo/15.0/odoo/fields.py", line 1243, in recompute
    self.compute_value(recs)
  File "/home/odoo/src/odoo/15.0/odoo/fields.py", line 1265, in compute_value
    records._compute_field_value(self)
  File "/home/odoo/src/odoo/15.0/odoo/models.py", line 4255, in _compute_field_value
    getattr(self, field.compute)()
  File "/home/odoo/runbot/extra/runbot/models/version.py", line 36, in _compute_version_number
    version.number = '.'.join([elem.zfill(2) for elem in re.sub(r'[^0-9\.]', '', version.name).split('.')])
  File "/usr/lib/python3.8/re.py", line 210, in sub
    return _compile(pattern, flags).sub(repl, string, count)
Exception

The above exception was the direct cause of the following exception:

Traceback (most recent call last):
  File "/home/odoo/src/odoo/15.0/odoo/http.py", line 643, in _handle_exception
    return super(JsonRequest, self)._handle_exception(exception)
  File "/home/odoo/src/odoo/15.0/odoo/http.py", line 301, in _handle_exception
    raise exception.with_traceback(None) from new_cause
TypeError: expected string or bytes-like object
2022-10-21 11:59:55 +02:00

106 lines
4.8 KiB
Python

import logging
import re
from odoo import models, fields, api, tools
_logger = logging.getLogger(__name__)
class Version(models.Model):
_name = 'runbot.version'
_description = "Version"
_order = 'sequence desc, number desc,id'
name = fields.Char('Version name')
number = fields.Char('Version number', compute='_compute_version_number', store=True, help="Usefull to sort by version")
sequence = fields.Integer('sequence')
is_major = fields.Char('Is major version', compute='_compute_version_number', store=True)
base_bundle_id = fields.Many2one('runbot.bundle', compute='_compute_base_bundle_id')
previous_major_version_id = fields.Many2one('runbot.version', compute='_compute_version_relations')
intermediate_version_ids = fields.Many2many('runbot.version', compute='_compute_version_relations')
next_major_version_id = fields.Many2one('runbot.version', compute='_compute_version_relations')
next_intermediate_version_ids = fields.Many2many('runbot.version', compute='_compute_version_relations')
dockerfile_id = fields.Many2one('runbot.dockerfile', default=lambda self: self.env.ref('runbot.docker_default', raise_if_not_found=False))
@api.depends('name')
def _compute_version_number(self):
for version in self:
if version.name == 'master':
version.number = '~'
version.is_major = False
else:
# max version number with this format: 99.99
version.number = '.'.join([elem.zfill(2) for elem in re.sub(r'[^0-9\.]', '', version.name or '').split('.')])
version.is_major = all(elem == '00' for elem in version.number.split('.')[1:])
@api.model_create_multi
def create(self, vals_list):
model = self.browse()
model._get_id.clear_cache(model)
return super().create(vals_list)
def _get(self, name):
return self.browse(self._get_id(name))
@tools.ormcache('name')
def _get_id(self, name):
version = self.search([('name', '=', name)])
if not version:
version = self.create({
'name': name,
})
return version.id
@api.depends('is_major', 'number')
def _compute_version_relations(self):
all_versions = self.search([], order='sequence, number')
for version in self:
version.previous_major_version_id = next(
(
v
for v in reversed(all_versions)
if v.is_major and v.number < version.number and v.sequence <= version.sequence # TODO FIXME, make version comparable?
), self.browse())
if version.previous_major_version_id:
version.intermediate_version_ids = all_versions.filtered(
lambda v, current=version: v.number > current.previous_major_version_id.number and v.number < current.number and v.sequence <= current.sequence and v.sequence >= current.previous_major_version_id.sequence
)
else:
version.intermediate_version_ids = all_versions.filtered(
lambda v, current=version: v.number < current.number and v.sequence <= current.sequence
)
version.next_major_version_id = next(
(
v
for v in all_versions
if (v.is_major or v.name == 'master') and v.number > version.number and v.sequence >= version.sequence
), self.browse())
if version.next_major_version_id:
version.next_intermediate_version_ids = all_versions.filtered(
lambda v, current=version: v.number < current.next_major_version_id.number and v.number > current.number and v.sequence <= current.next_major_version_id.sequence and v.sequence >= current.sequence
)
else:
version.next_intermediate_version_ids = all_versions.filtered(
lambda v, current=version: v.number > current.number and v.sequence >= current.sequence
)
# @api.depends('base_bundle_id.is_base', 'base_bundle_id.version_id', 'base_bundle_id.project_id')
@api.depends_context('project_id')
def _compute_base_bundle_id(self):
project_id = self.env.context.get('project_id')
if not project_id:
_logger.warning("_compute_base_bundle_id: no project_id in context")
project_id = self.env.ref('runbot.main_project').id
bundles = self.env['runbot.bundle'].search([
('version_id', 'in', self.ids),
('is_base', '=', True),
('project_id', '=', project_id)
])
bundle_by_version = {bundle.version_id.id: bundle for bundle in bundles}
for version in self:
version.base_bundle_id = bundle_by_version.get(version.id)