odoo-addon-odoo-project-stat 16.0.1.0.1.1__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,104 @@
1
+ .. image:: https://odoo-community.org/readme-banner-image
2
+ :target: https://odoo-community.org/get-involved?utm_source=readme
3
+ :alt: Odoo Community Association
4
+
5
+ ==================
6
+ Odoo Project Stats
7
+ ==================
8
+
9
+ ..
10
+ !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
11
+ !! This file is generated by oca-gen-addon-readme !!
12
+ !! changes will be overwritten. !!
13
+ !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
14
+ !! source digest: sha256:baa0b975fb4f8145367be53b53084d70771ccdcf9846a39db1dc337a6d1d111e
15
+ !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
16
+
17
+ .. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png
18
+ :target: https://odoo-community.org/page/development-status
19
+ :alt: Beta
20
+ .. |badge2| image:: https://img.shields.io/badge/license-AGPL--3-blue.png
21
+ :target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
22
+ :alt: License: AGPL-3
23
+ .. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fmodule--composition--analysis-lightgray.png?logo=github
24
+ :target: https://github.com/OCA/module-composition-analysis/tree/16.0/odoo_project_stat
25
+ :alt: OCA/module-composition-analysis
26
+ .. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png
27
+ :target: https://translation.odoo-community.org/projects/module-composition-analysis-16-0/module-composition-analysis-16-0-odoo_project_stat
28
+ :alt: Translate me on Weblate
29
+ .. |badge5| image:: https://img.shields.io/badge/runboat-Try%20me-875A7B.png
30
+ :target: https://runboat.odoo-community.org/builds?repo=OCA/module-composition-analysis&target_branch=16.0
31
+ :alt: Try me on Runboat
32
+
33
+ |badge1| |badge2| |badge3| |badge4| |badge5|
34
+
35
+ This module allows to generate and render some pie charts for your Odoo
36
+ projects.
37
+
38
+ It consists of two pie charts on the project form view:
39
+
40
+ - number of installed modules
41
+ - number of lines of code
42
+
43
+ Both measures are split based on some criteria (configurable), by
44
+ default:
45
+
46
+ - Odoo CE
47
+ - Odoo Enterprise
48
+ - OCA
49
+ - Community (not OCA)
50
+ - Specific code
51
+ - Misc. (everything not part of other criteria above)
52
+
53
+ Each criteria having a rendering color used on the charts.
54
+
55
+ |Project Pie Charts|
56
+
57
+ .. |Project Pie Charts| image:: https://raw.githubusercontent.com/OCA/module-composition-analysis/16.0/odoo_project_stat/static/img/project_pie_charts.png
58
+
59
+ **Table of contents**
60
+
61
+ .. contents::
62
+ :local:
63
+
64
+ Bug Tracker
65
+ ===========
66
+
67
+ Bugs are tracked on `GitHub Issues <https://github.com/OCA/module-composition-analysis/issues>`_.
68
+ In case of trouble, please check there if your issue has already been reported.
69
+ If you spotted it first, help us to smash it by providing a detailed and welcomed
70
+ `feedback <https://github.com/OCA/module-composition-analysis/issues/new?body=module:%20odoo_project_stat%0Aversion:%2016.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
71
+
72
+ Do not contact contributors directly about support or help with technical issues.
73
+
74
+ Credits
75
+ =======
76
+
77
+ Authors
78
+ -------
79
+
80
+ * Camptocamp
81
+
82
+ Contributors
83
+ ------------
84
+
85
+ - Camptocamp
86
+
87
+ - Sébastien Alix <seb@usr-src.org>
88
+
89
+ Maintainers
90
+ -----------
91
+
92
+ This module is maintained by the OCA.
93
+
94
+ .. image:: https://odoo-community.org/logo.png
95
+ :alt: Odoo Community Association
96
+ :target: https://odoo-community.org
97
+
98
+ OCA, or the Odoo Community Association, is a nonprofit organization whose
99
+ mission is to support the collaborative development of Odoo features and
100
+ promote its widespread use.
101
+
102
+ This module is part of the `OCA/module-composition-analysis <https://github.com/OCA/module-composition-analysis/tree/16.0/odoo_project_stat>`_ project on GitHub.
103
+
104
+ You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.
@@ -0,0 +1,2 @@
1
+ from . import models
2
+ from . import wizards
@@ -0,0 +1,23 @@
1
+ # Copyright 2023 Camptocamp SA
2
+ # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl)
3
+ {
4
+ "name": "Odoo Project Stats",
5
+ "summary": "Get some stats about your Odoo Projects.",
6
+ "version": "16.0.1.0.1",
7
+ "category": "Tools",
8
+ "author": "Camptocamp, Odoo Community Association (OCA)",
9
+ "website": "https://github.com/OCA/module-composition-analysis",
10
+ "data": [
11
+ "security/ir.model.access.csv",
12
+ "data/odoo_project_stat_config.xml",
13
+ "views/odoo_project.xml",
14
+ "views/odoo_project_stat_config.xml",
15
+ ],
16
+ "installable": True,
17
+ "depends": [
18
+ "odoo_project",
19
+ # OCA/web
20
+ "web_widget_plotly_chart",
21
+ ],
22
+ "license": "AGPL-3",
23
+ }
@@ -0,0 +1,77 @@
1
+ <?xml version="1.0" encoding="utf-8" ?>
2
+ <!-- Copyright 2024 Camptocamp SA
3
+ License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
4
+ <odoo noupdate="1">
5
+
6
+ <record id="odoo_project_stat_config_odoo_ce" model="odoo.project.stat.config">
7
+ <field name="sequence" eval="1" />
8
+ <field name="name">Odoo CE</field>
9
+ <field name="color">#985184</field>
10
+ <field
11
+ name="domain"
12
+ eval="[
13
+ ('is_standard', '=', True),
14
+ ('is_enterprise', '=', False),
15
+ ('org_id', '=', ref('odoo_repository.odoo_repository_org_odoo')),
16
+ ]"
17
+ />
18
+ </record>
19
+
20
+ <record
21
+ id="odoo_project_stat_config_odoo_enterprise"
22
+ model="odoo.project.stat.config"
23
+ >
24
+ <field name="sequence" eval="2" />
25
+ <field name="name">Odoo Enterprise</field>
26
+ <field name="color">#1ad3bb</field>
27
+ <field
28
+ name="domain"
29
+ eval="[
30
+ ('is_standard', '=', True),
31
+ ('is_enterprise', '=', True),
32
+ ('org_id', '=', ref('odoo_repository.odoo_repository_org_odoo')),
33
+ ]"
34
+ />
35
+ </record>
36
+
37
+ <record id="odoo_project_stat_config_oca" model="odoo.project.stat.config">
38
+ <field name="sequence" eval="3" />
39
+ <field name="name">OCA</field>
40
+ <field name="color">#a4b0f1</field>
41
+ <field
42
+ name="domain"
43
+ eval="[
44
+ ('org_id', '=', ref('odoo_repository.odoo_repository_org_oca')),
45
+ ]"
46
+ />
47
+ </record>
48
+
49
+ <record id="odoo_project_stat_config_community" model="odoo.project.stat.config">
50
+ <field name="sequence" eval="4" />
51
+ <field name="name">Community</field>
52
+ <field name="color">#c3ccfc</field>
53
+ <field
54
+ name="domain"
55
+ eval="[
56
+ ('is_community', '=', True),
57
+ ('specific', '=', False),
58
+ ('org_id', 'not in', (ref('odoo_repository.odoo_repository_org_odoo'), ref('odoo_repository.odoo_repository_org_oca'))),
59
+ ]"
60
+ />
61
+ </record>
62
+
63
+ <record id="odoo_project_stat_config_specific" model="odoo.project.stat.config">
64
+ <field name="sequence" eval="5" />
65
+ <field name="name">Specific</field>
66
+ <field name="color">#ff680a</field>
67
+ <field name="domain" eval="[('specific', '=', True)]" />
68
+ </record>
69
+
70
+ <record id="odoo_project_stat_config_others" model="odoo.project.stat.config">
71
+ <field name="sequence" eval="6" />
72
+ <field name="name">Misc.</field>
73
+ <field name="color">#f3f4f6</field>
74
+ <field name="residual" eval="True" />
75
+ </record>
76
+
77
+ </odoo>
@@ -0,0 +1,180 @@
1
+ # Translation of Odoo Server.
2
+ # This file contains the translation of the following modules:
3
+ # * odoo_project_stat
4
+ #
5
+ msgid ""
6
+ msgstr ""
7
+ "Project-Id-Version: Odoo Server 16.0\n"
8
+ "Report-Msgid-Bugs-To: \n"
9
+ "Last-Translator: \n"
10
+ "Language-Team: \n"
11
+ "MIME-Version: 1.0\n"
12
+ "Content-Type: text/plain; charset=UTF-8\n"
13
+ "Content-Transfer-Encoding: \n"
14
+ "Plural-Forms: \n"
15
+
16
+ #. module: odoo_project_stat
17
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__color
18
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__color
19
+ msgid "Color"
20
+ msgstr ""
21
+
22
+ #. module: odoo_project_stat
23
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__create_uid
24
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__create_uid
25
+ msgid "Created by"
26
+ msgstr ""
27
+
28
+ #. module: odoo_project_stat
29
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__create_date
30
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__create_date
31
+ msgid "Created on"
32
+ msgstr ""
33
+
34
+ #. module: odoo_project_stat
35
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__date
36
+ msgid "Date"
37
+ msgstr ""
38
+
39
+ #. module: odoo_project_stat
40
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__display_name
41
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__display_name
42
+ msgid "Display Name"
43
+ msgstr ""
44
+
45
+ #. module: odoo_project_stat
46
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__domain
47
+ msgid "Domain"
48
+ msgstr ""
49
+
50
+ #. module: odoo_project_stat
51
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__id
52
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__id
53
+ msgid "ID"
54
+ msgstr ""
55
+
56
+ #. module: odoo_project_stat
57
+ #: model:ir.model,name:odoo_project_stat.model_odoo_project_import_modules
58
+ msgid "Import modules for an Odoo project"
59
+ msgstr ""
60
+
61
+ #. module: odoo_project_stat
62
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat____last_update
63
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config____last_update
64
+ msgid "Last Modified on"
65
+ msgstr ""
66
+
67
+ #. module: odoo_project_stat
68
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__write_uid
69
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__write_uid
70
+ msgid "Last Updated by"
71
+ msgstr ""
72
+
73
+ #. module: odoo_project_stat
74
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__write_date
75
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__write_date
76
+ msgid "Last Updated on"
77
+ msgstr ""
78
+
79
+ #. module: odoo_project_stat
80
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project__chart_sloc
81
+ msgid "Lines of Code Chart"
82
+ msgstr ""
83
+
84
+ #. module: odoo_project_stat
85
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__sloc
86
+ msgid "Lines of code"
87
+ msgstr ""
88
+
89
+ #. module: odoo_project_stat
90
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project__chart_modules_count
91
+ msgid "Modules Chart"
92
+ msgstr ""
93
+
94
+ #. module: odoo_project_stat
95
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__modules_count
96
+ msgid "Modules Count"
97
+ msgstr ""
98
+
99
+ #. module: odoo_project_stat
100
+ #: model:ir.model,name:odoo_project_stat.model_odoo_project
101
+ msgid "Odoo Project"
102
+ msgstr ""
103
+
104
+ #. module: odoo_project_stat
105
+ #: model:ir.model,name:odoo_project_stat.model_odoo_project_stat_config
106
+ msgid "Odoo Project Stat Config"
107
+ msgstr ""
108
+
109
+ #. module: odoo_project_stat
110
+ #: model:ir.model,name:odoo_project_stat.model_odoo_project_stat
111
+ msgid "Odoo Project Stats"
112
+ msgstr ""
113
+
114
+ #. module: odoo_project_stat
115
+ #: model:ir.model.constraint,message:odoo_project_stat.constraint_odoo_project_stat_config_residual_uniq
116
+ msgid "Only one configuration should exist for residual modules."
117
+ msgstr ""
118
+
119
+ #. module: odoo_project_stat
120
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__odoo_project_id
121
+ msgid "Project"
122
+ msgstr ""
123
+
124
+ #. module: odoo_project_stat
125
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__config_id
126
+ msgid "Project Stat Configuration"
127
+ msgstr ""
128
+
129
+ #. module: odoo_project_stat
130
+ #: model:ir.actions.act_window,name:odoo_project_stat.odoo_project_stat_config_action
131
+ #: model:ir.ui.menu,name:odoo_project_stat.odoo_project_stat_config_menu
132
+ msgid "Project Stats"
133
+ msgstr ""
134
+
135
+ #. module: odoo_project_stat
136
+ #: model_terms:ir.ui.view,arch_db:odoo_project_stat.odoo_project_view_form
137
+ msgid "Refresh"
138
+ msgstr ""
139
+
140
+ #. module: odoo_project_stat
141
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__residual
142
+ msgid "Residual"
143
+ msgstr ""
144
+
145
+ #. module: odoo_project_stat
146
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__sequence
147
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__sequence
148
+ msgid "Sequence"
149
+ msgstr ""
150
+
151
+ #. module: odoo_project_stat
152
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project__stats_ids
153
+ #: model_terms:ir.ui.view,arch_db:odoo_project_stat.odoo_project_view_form
154
+ msgid "Stats"
155
+ msgstr ""
156
+
157
+ #. module: odoo_project_stat
158
+ #: model:ir.model.constraint,message:odoo_project_stat.constraint_odoo_project_stat_odoo_project_config_date_uniq
159
+ msgid "This project stats record already exists."
160
+ msgstr ""
161
+
162
+ #. module: odoo_project_stat
163
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat__name
164
+ #: model:ir.model.fields,field_description:odoo_project_stat.field_odoo_project_stat_config__name
165
+ msgid "Title"
166
+ msgstr ""
167
+
168
+ #. module: odoo_project_stat
169
+ #: model_terms:ir.ui.view,arch_db:odoo_project_stat.odoo_project_view_form
170
+ msgid ""
171
+ "Useful to get accurate figures if modules have been updated in their "
172
+ "respective repositories."
173
+ msgstr ""
174
+
175
+ #. module: odoo_project_stat
176
+ #: model:ir.model.fields,help:odoo_project_stat.field_odoo_project_stat_config__residual
177
+ msgid ""
178
+ "Will catch all modules that doesn't match other rules (domain field will be "
179
+ "ignored if enabled)."
180
+ msgstr ""
@@ -0,0 +1,3 @@
1
+ from . import odoo_project_stat_config
2
+ from . import odoo_project_stat
3
+ from . import odoo_project
@@ -0,0 +1,109 @@
1
+ # Copyright 2024 Camptocamp SA
2
+ # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl)
3
+
4
+ import plotly.graph_objects as go
5
+ from plotly.offline import plot
6
+
7
+ from odoo import fields, models
8
+
9
+
10
+ class OdooProject(models.Model):
11
+ _inherit = "odoo.project"
12
+
13
+ stats_ids = fields.One2many(
14
+ comodel_name="odoo.project.stat",
15
+ inverse_name="odoo_project_id",
16
+ )
17
+ chart_modules_count = fields.Text(
18
+ string="Modules Chart",
19
+ compute="_compute_charts",
20
+ )
21
+ chart_sloc = fields.Text(
22
+ string="Lines of Code Chart",
23
+ compute="_compute_charts",
24
+ )
25
+
26
+ def _get_last_stats(self):
27
+ self.ensure_one()
28
+ last_stat = self.env["odoo.project.stat"].search(
29
+ [("odoo_project_id", "=", self.id)],
30
+ order="date DESC",
31
+ limit=1,
32
+ )
33
+ if not last_stat:
34
+ return self.env["odoo.project.stat"].browse()
35
+ return self.env["odoo.project.stat"].search(
36
+ [("odoo_project_id", "=", self.id), ("date", "=", last_stat.date)],
37
+ order="sequence",
38
+ )
39
+
40
+ def _compute_charts(self):
41
+ for rec in self:
42
+ rec.chart_modules_count = rec.chart_sloc = False
43
+ stats = rec._get_last_stats()
44
+ labels = stats.mapped("name")
45
+ colors = stats.mapped("color")
46
+ count_values = stats.mapped("modules_count")
47
+ sloc_values = stats.mapped("sloc")
48
+ # Modules Count Pie Chart
49
+ if count_values:
50
+ count_chart = go.Figure(
51
+ data=[
52
+ go.Pie(
53
+ labels=labels,
54
+ values=count_values,
55
+ )
56
+ ]
57
+ )
58
+ self._chart_update_layout(count_chart, colors=colors, title="Modules")
59
+ rec.chart_modules_count = plot(
60
+ count_chart,
61
+ include_plotlyjs=False,
62
+ output_type="div",
63
+ config={"displaylogo": False},
64
+ )
65
+ # SLOC Pie Chart
66
+ if sloc_values:
67
+ sloc_chart = go.Figure(
68
+ data=[
69
+ go.Pie(
70
+ labels=labels,
71
+ values=sloc_values,
72
+ )
73
+ ]
74
+ )
75
+ self._chart_update_layout(
76
+ sloc_chart, colors=colors, title="Lines of Code"
77
+ )
78
+ rec.chart_sloc = plot(
79
+ sloc_chart,
80
+ include_plotlyjs=False,
81
+ output_type="div",
82
+ config={"displaylogo": False},
83
+ )
84
+
85
+ def _chart_update_layout(self, chart, colors=None, title=None):
86
+ chart.update_traces(
87
+ textinfo="percent+label",
88
+ textposition="inside",
89
+ hole=0.35,
90
+ sort=False,
91
+ direction="clockwise",
92
+ rotation=180,
93
+ )
94
+ if colors:
95
+ chart.update_traces(marker=dict(colors=colors))
96
+ if title:
97
+ chart.update_layout(title_text=title, title_x=0.5, title_y=0.5)
98
+ chart.update_layout(
99
+ margin=dict(l=0, r=0, t=20, b=20),
100
+ showlegend=False,
101
+ autosize=False,
102
+ width=450,
103
+ height=450,
104
+ )
105
+
106
+ def action_generate_stats(self):
107
+ for rec in self:
108
+ self.env["odoo.project.stat"].sudo()._generate_stats(rec.id)
109
+ return True
@@ -0,0 +1,114 @@
1
+ # Copyright 2024 Camptocamp SA
2
+ # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl)
3
+
4
+ from odoo import fields, models
5
+ from odoo.tools.safe_eval import safe_eval
6
+
7
+
8
+ class OdooProjectStat(models.Model):
9
+ _name = "odoo.project.stat"
10
+ _description = "Odoo Project Stats"
11
+ _rec_name = "name"
12
+ _order = "odoo_project_id, date, sequence, name"
13
+
14
+ odoo_project_id = fields.Many2one(
15
+ comodel_name="odoo.project",
16
+ ondelete="cascade",
17
+ string="Project",
18
+ index=True,
19
+ required=True,
20
+ readonly=True,
21
+ )
22
+ config_id = fields.Many2one(
23
+ comodel_name="odoo.project.stat.config",
24
+ ondelete="restrict",
25
+ string="Project Stat Configuration",
26
+ required=True,
27
+ readonly=True,
28
+ )
29
+ date = fields.Date(required=True, index=True)
30
+ sequence = fields.Integer(related="config_id.sequence", store=True)
31
+ name = fields.Char(related="config_id.name", store=True)
32
+ color = fields.Char(related="config_id.color")
33
+ modules_count = fields.Integer(readonly=True)
34
+ sloc = fields.Integer(string="Lines of code", readonly=True)
35
+
36
+ _sql_constraints = [
37
+ (
38
+ "odoo_project_config_date_uniq",
39
+ "UNIQUE (odoo_project_id, config_id, date)",
40
+ "This project stats record already exists.",
41
+ ),
42
+ ]
43
+
44
+ def _get_stats(self, odoo_project, date=None, config=None, limit=None):
45
+ domain = [("odoo_project_id", "=", odoo_project.id)]
46
+ if date:
47
+ domain.append(("date", "=", date))
48
+ if config:
49
+ domain.append(("config_id", "=", config.id))
50
+ return self.search(domain, limit=limit)
51
+
52
+ def _generate_stats(self, odoo_project_id):
53
+ """Generate the stats for a given `odoo_project_id`."""
54
+ odoo_project = self.env["odoo.project"].browse(odoo_project_id).exists()
55
+ odoo_project.ensure_one()
56
+ modules = odoo_project.project_module_ids
57
+ total_count = len(modules)
58
+ total_sloc = (
59
+ sum(modules.mapped("sloc_python"))
60
+ + sum(modules.mapped("sloc_xml"))
61
+ + sum(modules.mapped("sloc_js"))
62
+ + sum(modules.mapped("sloc_css"))
63
+ )
64
+ # Clean up stats of today if any
65
+ today = fields.Date.today()
66
+ existing_stats = self._get_stats(odoo_project, date=today)
67
+ existing_stats.sudo().unlink()
68
+ # Create or update existing stat record
69
+ configs = self.env["odoo.project.stat.config"].search([])
70
+ for config in configs:
71
+ stat = self._get_stats(odoo_project, date=today, config=config, limit=1)
72
+ values = self._generate_stat_values(odoo_project, config, total_count)
73
+ if stat:
74
+ stat.sudo().write(values)
75
+ else:
76
+ self.sudo().create(values)
77
+ stats = self._get_stats(odoo_project, date=today)
78
+ stat_residual = stats.filtered(lambda o: o.config_id.residual)
79
+ if stat_residual:
80
+ other_stats = stats - stat_residual
81
+ stat_residual.sudo().write(
82
+ {
83
+ "modules_count": (
84
+ total_count - sum(other_stats.mapped("modules_count"))
85
+ ),
86
+ "sloc": (total_sloc - sum(other_stats.mapped("sloc"))),
87
+ }
88
+ )
89
+ return True
90
+
91
+ def _generate_stat_values(self, odoo_project, config, total_count):
92
+ # Counter
93
+ if config.residual:
94
+ modules_count = 0
95
+ sloc = 0
96
+ else:
97
+ domain = safe_eval(config.domain)
98
+ # FIXME: use odoo.osv.expression.AND
99
+ domain.append(("odoo_project_id", "=", odoo_project.id))
100
+ modules = self.env["odoo.project.module"].search(domain)
101
+ modules_count = len(modules)
102
+ sloc = (
103
+ sum(modules.mapped("sloc_python"))
104
+ + sum(modules.mapped("sloc_xml"))
105
+ + sum(modules.mapped("sloc_js"))
106
+ + sum(modules.mapped("sloc_css"))
107
+ )
108
+ return {
109
+ "odoo_project_id": odoo_project.id,
110
+ "config_id": config.id,
111
+ "date": fields.Date.today(),
112
+ "modules_count": modules_count,
113
+ "sloc": sloc,
114
+ }
@@ -0,0 +1,29 @@
1
+ # Copyright 2024 Camptocamp SA
2
+ # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl)
3
+
4
+ from odoo import fields, models
5
+
6
+
7
+ class OdooProjectStatConfig(models.Model):
8
+ _name = "odoo.project.stat.config"
9
+ _description = "Odoo Project Stat Config"
10
+ _order = "sequence, name"
11
+
12
+ sequence = fields.Integer(default=10)
13
+ name = fields.Char(string="Title", required=True)
14
+ color = fields.Char(required=True)
15
+ residual = fields.Boolean(
16
+ help=(
17
+ "Will catch all modules that doesn't match other rules "
18
+ "(domain field will be ignored if enabled)."
19
+ )
20
+ )
21
+ domain = fields.Char()
22
+
23
+ _sql_constraints = [
24
+ (
25
+ "residual_uniq",
26
+ "UNIQUE (residual)",
27
+ "Only one configuration should exist for residual modules.",
28
+ ),
29
+ ]
@@ -0,0 +1,2 @@
1
+ - Camptocamp
2
+ - Sébastien Alix \<seb@usr-src.org\>
@@ -0,0 +1,19 @@
1
+ This module allows to generate and render some pie charts for your Odoo projects.
2
+
3
+ It consists of two pie charts on the project form view:
4
+
5
+ - number of installed modules
6
+ - number of lines of code
7
+
8
+ Both measures are split based on some criteria (configurable), by default:
9
+ - Odoo CE
10
+ - Odoo Enterprise
11
+ - OCA
12
+ - Community (not OCA)
13
+ - Specific code
14
+ - Misc. (everything not part of other criteria above)
15
+
16
+ Each criteria having a rendering color used on the charts.
17
+
18
+ ![Project Pie Charts](./static/img/project_pie_charts.png)
19
+
@@ -0,0 +1,4 @@
1
+ id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
2
+ access_odoo_project_stat_config_user,odoo_project_stat_config_user,model_odoo_project_stat_config,odoo_repository.group_odoo_repository_user,1,0,0,0
3
+ access_odoo_project_stat_config_manager,odoo_project_stat_config_manager,model_odoo_project_stat_config,odoo_repository.group_odoo_repository_manager,1,1,1,1
4
+ access_odoo_project_stat_user,odoo_project_stat_user,model_odoo_project_stat,odoo_repository.group_odoo_repository_user,1,0,0,0