mirror of
https://github.com/odoo/runbot.git
synced 2025-03-30 23:05:44 +07:00
[IMP] runbot: add a codeowner model
As a custom codeowner system was successfully implemented in a python step on our runbot instance, it's now time to have a real model for that. This commit adds a skeleton Codeowner model in order to be used for a basic usage. This should be improved in the future after some battle testing.
This commit is contained in:
parent
89dcb52215
commit
91efbec943
@ -41,6 +41,7 @@
|
|||||||
'views/build_error_views.xml',
|
'views/build_error_views.xml',
|
||||||
'views/build_views.xml',
|
'views/build_views.xml',
|
||||||
'views/bundle_views.xml',
|
'views/bundle_views.xml',
|
||||||
|
'views/codeowner_views.xml',
|
||||||
'views/commit_views.xml',
|
'views/commit_views.xml',
|
||||||
'views/config_views.xml',
|
'views/config_views.xml',
|
||||||
'views/dashboard_views.xml',
|
'views/dashboard_views.xml',
|
||||||
|
@ -6,6 +6,7 @@ from . import build
|
|||||||
from . import build_config
|
from . import build_config
|
||||||
from . import build_error
|
from . import build_error
|
||||||
from . import bundle
|
from . import bundle
|
||||||
|
from . import codeowner
|
||||||
from . import commit
|
from . import commit
|
||||||
from . import database
|
from . import database
|
||||||
from . import dockerfile
|
from . import dockerfile
|
||||||
@ -22,5 +23,6 @@ from . import upgrade
|
|||||||
from . import user
|
from . import user
|
||||||
from . import version
|
from . import version
|
||||||
|
|
||||||
|
# those imports have to be at the end otherwise the sql view cannot be initialised
|
||||||
from . import build_stat
|
from . import build_stat
|
||||||
from . import build_stat_regex
|
from . import build_stat_regex
|
||||||
|
30
runbot/models/codeowner.py
Normal file
30
runbot/models/codeowner.py
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import ast
|
||||||
|
import re
|
||||||
|
|
||||||
|
from odoo import models, fields, api
|
||||||
|
from odoo.exceptions import ValidationError
|
||||||
|
|
||||||
|
|
||||||
|
class Codeowner(models.Model):
|
||||||
|
_name = 'runbot.codeowner'
|
||||||
|
_description = "Notify github teams based on filenames regex"
|
||||||
|
_inherit = "mail.thread"
|
||||||
|
|
||||||
|
project_id = fields.Many2one('runbot.project', required=True)
|
||||||
|
regex = fields.Char('Regular Expression', help='Regex to match full file paths', required=True, tracking=True)
|
||||||
|
github_teams = fields.Char(help='Comma separated list of github teams to notify', required=True, tracking=True)
|
||||||
|
team_id = fields.Many2one('runbot.team', help='Not mandatory runbot team')
|
||||||
|
version_domain = fields.Char('Version Domain', help='Codeowner only applies to the filtered versions')
|
||||||
|
|
||||||
|
@api.constrains('regex')
|
||||||
|
def _validate_regex(self):
|
||||||
|
for rec in self:
|
||||||
|
try:
|
||||||
|
r = re.compile(rec.regex)
|
||||||
|
except re.error as e:
|
||||||
|
raise ValidationError("Unable to compile regular expression: %s" % e)
|
||||||
|
|
||||||
|
def _get_version_domain(self):
|
||||||
|
""" Helper to get the evaluated version domain """
|
||||||
|
self.ensure_one()
|
||||||
|
return ast.eval(self.version_domain) if self.version_domain else []
|
@ -111,3 +111,6 @@ access_runbot_upgrade_exception_admin,access_runbot_upgrade_exception_admin,runb
|
|||||||
|
|
||||||
access_runbot_dockerfile_user,access_runbot_dockerfile_user,runbot.model_runbot_dockerfile,runbot.group_user,1,0,0,0
|
access_runbot_dockerfile_user,access_runbot_dockerfile_user,runbot.model_runbot_dockerfile,runbot.group_user,1,0,0,0
|
||||||
access_runbot_dockerfile_admin,access_runbot_dockerfile_admin,runbot.model_runbot_dockerfile,runbot.group_runbot_admin,1,1,1,1
|
access_runbot_dockerfile_admin,access_runbot_dockerfile_admin,runbot.model_runbot_dockerfile,runbot.group_runbot_admin,1,1,1,1
|
||||||
|
|
||||||
|
access_runbot_codeowner_admin,runbot_codeowner_admin,runbot.model_runbot_codeowner,runbot.group_runbot_admin,1,1,1,1
|
||||||
|
access_runbot_codeowner_user,runbot_codeowner_user,runbot.model_runbot_codeowner,group_user,1,0,0,0
|
||||||
|
|
@ -191,3 +191,35 @@ class TestBuildError(RunbotCase):
|
|||||||
})
|
})
|
||||||
|
|
||||||
self.assertEqual(dashboard.build_ids, failed_build)
|
self.assertEqual(dashboard.build_ids, failed_build)
|
||||||
|
|
||||||
|
class TestCodeOwner(RunbotCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super().setUp()
|
||||||
|
self.cow_deb = self.env['runbot.codeowner'].create({
|
||||||
|
'project_id' : self.project.id,
|
||||||
|
'github_teams': 'runbot',
|
||||||
|
'regex': '.*debian.*'
|
||||||
|
})
|
||||||
|
|
||||||
|
self.cow_web = self.env['runbot.codeowner'].create({
|
||||||
|
'project_id' : self.project.id,
|
||||||
|
'github_teams': 'website',
|
||||||
|
'regex': '.*website.*'
|
||||||
|
})
|
||||||
|
|
||||||
|
self.cow_crm = self.env['runbot.codeowner'].create({
|
||||||
|
'project_id' : self.project.id,
|
||||||
|
'github_teams': 'crm',
|
||||||
|
'regex': '.*crm.*'
|
||||||
|
})
|
||||||
|
|
||||||
|
self.cow_all = self.cow_deb | self.cow_web | self.cow_crm
|
||||||
|
|
||||||
|
def test_codeowner_invalid_regex(self):
|
||||||
|
with self.assertRaises(ValidationError):
|
||||||
|
self.env['runbot.codeowner'].create({
|
||||||
|
'project_id': self.project.id,
|
||||||
|
'regex': '*debian.*',
|
||||||
|
'github_teams': 'rd-test'
|
||||||
|
})
|
||||||
|
53
runbot/views/codeowner_views.xml
Normal file
53
runbot/views/codeowner_views.xml
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
<odoo>
|
||||||
|
<data>
|
||||||
|
<record id="codeowner_form" model="ir.ui.view">
|
||||||
|
<field name="name">runbot.codeowner.form</field>
|
||||||
|
<field name="model">runbot.codeowner</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form string="Host">
|
||||||
|
<sheet>
|
||||||
|
<group>
|
||||||
|
<field name="project_id"/>
|
||||||
|
<field name="team_id"/>
|
||||||
|
<field name="github_teams"/>
|
||||||
|
<field name="regex"/>
|
||||||
|
<field name="version_domain" widget="domain" options="{'model': 'runbot.version', 'in_dialog': True, 'operators': ['in','=', '<', '>']}"/>
|
||||||
|
</group>
|
||||||
|
</sheet>
|
||||||
|
<div class="oe_chatter">
|
||||||
|
<field name="message_follower_ids" widget="mail_followers"/>
|
||||||
|
<field name="message_ids" widget="mail_thread"/>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="view_codeowner_tree" model="ir.ui.view">
|
||||||
|
<field name="name">runbot.codeowner.tree</field>
|
||||||
|
<field name="model">runbot.codeowner</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<tree string="Codeowners">
|
||||||
|
<field name="project_id"/>
|
||||||
|
<field name="team_id"/>
|
||||||
|
<field name="version_domain"/>
|
||||||
|
<field name="regex"/>
|
||||||
|
<field name="github_teams"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="open_view_codeowner_tree" model="ir.actions.act_window">
|
||||||
|
<field name="name">Codeowner</field>
|
||||||
|
<field name="res_model">runbot.codeowner</field>
|
||||||
|
<field name="view_mode">tree,form</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<menuitem
|
||||||
|
name="CodeOwners"
|
||||||
|
id="runbot_menu_codeowner_tree"
|
||||||
|
parent="runbot_menu_manage_errors"
|
||||||
|
sequence="90"
|
||||||
|
action="open_view_codeowner_tree"
|
||||||
|
/>
|
||||||
|
</data>
|
||||||
|
</odoo>
|
Loading…
Reference in New Issue
Block a user