Browse Source

[IMP]Improved Pep8.

pull/45/head
Mustufa Kantawala 5 years ago
parent
commit
c212850112
  1. 1
      oca_dependencies.txt
  2. 29
      privacy_partner_report/__manifest__.py
  3. 46
      privacy_partner_report/report/privacy_partner_xlsx.py
  4. 15
      privacy_partner_report/views/privacy_menu_view.xml
  5. 194
      privacy_partner_report/wizard/privacy_report_partner.py
  6. 84
      privacy_partner_report/wizard/privacy_report_partner_wizard.xml

1
oca_dependencies.txt

@ -1,2 +1 @@
reporting-engine reporting-engine

29
privacy_partner_report/__manifest__.py

@ -1,20 +1,19 @@
# Copyright 2018 Eficent Business and IT Consulting Services S.L. # Copyright 2018 Eficent Business and IT Consulting Services S.L.
# License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html # License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html
{ {
'name': 'Privacy Partner Report',
'version': '13.0.1.0.0',
'category': 'GDPR',
'summary': 'Show the transactions that a specific partner is involved in.',
'author': "Eficent, "
"Odoo Community Association (OCA)",
'website': 'https://github.com/OCA/data-protection/',
'license': 'AGPL-3',
'depends': ['privacy', 'report_xlsx'],
'data': [
'wizard/privacy_report_partner_wizard.xml',
'views/privacy_report.xml',
'views/privacy_menu_view.xml',
"name": "Privacy Partner Report",
"version": "13.0.1.0.0",
"category": "GDPR",
"summary": "Show the transactions that a specific partner is involved in.",
"author": "Eficent, " "Odoo Community Association (OCA)",
"website": "https://github.com/OCA/data-protection/",
"license": "AGPL-3",
"depends": ["privacy", "report_xlsx"],
"data": [
"wizard/privacy_report_partner_wizard.xml",
"views/privacy_report.xml",
"views/privacy_menu_view.xml",
], ],
'installable': True,
'maintainers': ['mreficent'],
"installable": True,
"maintainers": ["mreficent"],
} }

46
privacy_partner_report/report/privacy_partner_xlsx.py

@ -2,6 +2,7 @@
# License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html # License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html
import logging import logging
from odoo import models from odoo import models
from odoo.tools.translate import _ from odoo.tools.translate import _
@ -9,8 +10,8 @@ _logger = logging.getLogger(__name__)
class ReportPartnerXlsx(models.AbstractModel): class ReportPartnerXlsx(models.AbstractModel):
_name = 'report.privacy_partner_report.report_partner_xlsx'
_inherit = 'report.report_xlsx.abstract'
_name = "report.privacy_partner_report.report_partner_xlsx"
_inherit = "report.report_xlsx.abstract"
def _search_longest_row(self, tables): def _search_longest_row(self, tables):
res = 0 res = 0
@ -24,26 +25,26 @@ class ReportPartnerXlsx(models.AbstractModel):
def generate_xlsx_report(self, workbook, data, objects): def generate_xlsx_report(self, workbook, data, objects):
for o in objects: for o in objects:
report_data = o.compute_data_for_report(data) report_data = o.compute_data_for_report(data)
partner = report_data['form'].get('partner_id', False)
partner = self.env['res.partner'].sudo().browse(partner[0])
workbook.set_properties({
'comments': 'Created with Python and XlsxWriter from Odoo'})
sheet = workbook.add_worksheet(_('Partner Data'))
partner = report_data["form"].get("partner_id", False)
partner = self.env["res.partner"].sudo().browse(partner[0])
workbook.set_properties(
{"comments": "Created with Python and XlsxWriter from Odoo"}
)
sheet = workbook.add_worksheet(_("Partner Data"))
sheet.set_landscape() sheet.set_landscape()
sheet.fit_to_pages(1, 0) sheet.fit_to_pages(1, 0)
sheet.set_zoom(75) sheet.set_zoom(75)
sheet.set_column(0, self._search_longest_row(
report_data['tables']), 25)
sheet.set_column(0, self._search_longest_row(report_data["tables"]), 25)
title_style = workbook.add_format( title_style = workbook.add_format(
{'bold': True, 'bg_color': '#FFFFCC', 'border': 2})
sheet.set_row(0, None, None, {'collapsed': 1})
sheet.write_row(1, 0, ["Partner: " + partner.display_name],
title_style)
{"bold": True, "bg_color": "#FFFFCC", "border": 2}
)
sheet.set_row(0, None, None, {"collapsed": 1})
sheet.write_row(1, 0, ["Partner: " + partner.display_name], title_style)
i = 3 i = 3
first_row = i+2
for table in sorted(report_data['tables'].keys()):
for model in sorted(report_data['tables'][table].keys()):
rows = len(report_data['tables'][table][model])
first_row = i + 2
for table in sorted(report_data["tables"].keys()):
for model in sorted(report_data["tables"][table].keys()):
rows = len(report_data["tables"][table][model])
if rows: if rows:
style = workbook.add_format() style = workbook.add_format()
style.set_bold(True) style.set_bold(True)
@ -51,19 +52,18 @@ class ReportPartnerXlsx(models.AbstractModel):
sheet.write_row(i, 0, [model], style) sheet.write_row(i, 0, [model], style)
i += 1 i += 1
j = 0 j = 0
for column in report_data['tables'][table][model][0]:
for column in report_data["tables"][table][model][0]:
style = workbook.add_format() style = workbook.add_format()
style.set_bold(True) style.set_bold(True)
if j == 0: if j == 0:
style.set_left(1) style.set_left(1)
if j == len(report_data['tables'][
table][model][0]) - 1:
if j == len(report_data["tables"][table][model][0]) - 1:
style.set_right(1) style.set_right(1)
style.set_top(1) style.set_top(1)
style.set_bottom(1) style.set_bottom(1)
sheet.write_row(i, j, [column], style) sheet.write_row(i, j, [column], style)
j += 1 j += 1
for row in report_data['tables'][table][model]:
for row in report_data["tables"][table][model]:
i += 1 i += 1
j = 0 j = 0
for column in row: for column in row:
@ -77,7 +77,7 @@ class ReportPartnerXlsx(models.AbstractModel):
if row[column]: if row[column]:
sheet.write_row(i, j, [row[column]], style) sheet.write_row(i, j, [row[column]], style)
else: else:
sheet.write_row(i, j, [''], style)
sheet.write_row(i, j, [""], style)
j += 1 j += 1
i += 2 i += 2
first_row = i+2
first_row = i + 2

15
privacy_partner_report/views/privacy_menu_view.xml

@ -1,13 +1,12 @@
<!-- Copyright 2018 Eficent Business and IT Consulting Services S.L. <!-- Copyright 2018 Eficent Business and IT Consulting Services S.L.
License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl-3.0) --> License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl-3.0) -->
<odoo> <odoo>
<menuitem id="menu_privacy_report_privacy_report"
name="Partner Report"
parent="privacy.menu_data_protection_report"
groups="privacy.group_data_protection_user"
sequence="10"
action="privacy_partner_report.action_privacy_partner_menu"
<menuitem
id="menu_privacy_report_privacy_report"
name="Partner Report"
parent="privacy.menu_data_protection_report"
groups="privacy.group_data_protection_user"
sequence="10"
action="privacy_partner_report.action_privacy_partner_menu"
/> />
</odoo> </odoo>

194
privacy_partner_report/wizard/privacy_report_partner.py

@ -1,77 +1,70 @@
# Copyright 2018 Eficent Business and IT Consulting Services S.L. # Copyright 2018 Eficent Business and IT Consulting Services S.L.
# License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html # License AGPL-3 - See http://www.gnu.org/licenses/agpl-3.0.html
from odoo import api, fields, models, _
from odoo.exceptions import UserError
import ast import ast
from odoo import _, api, fields, models
from odoo.exceptions import UserError
class PrivacyPartnerReport(models.TransientModel): class PrivacyPartnerReport(models.TransientModel):
_name = "privacy.partner.report" _name = "privacy.partner.report"
_description = "Privacy Partner Report" _description = "Privacy Partner Report"
company_id = fields.Many2one( company_id = fields.Many2one(
comodel_name='res.company',
string='Company',
comodel_name="res.company",
string="Company",
required=True, required=True,
default=lambda self: self.env.user.company_id, default=lambda self: self.env.user.company_id,
) )
partner_id = fields.Many2one( partner_id = fields.Many2one(
comodel_name='res.partner',
string='Partner',
required=True,
comodel_name="res.partner", string="Partner", required=True,
) )
table_ids = fields.Many2many( table_ids = fields.Many2many(
comodel_name='privacy.partner.data',
string='Models with related partner data',
comodel_name="privacy.partner.data", string="Models with related partner data",
) )
@api.onchange('partner_id')
@api.onchange("partner_id")
def _onchange_partner_id(self): def _onchange_partner_id(self):
if self.partner_id: if self.partner_id:
data = self._get_tables_from_partner(self.partner_id) data = self._get_tables_from_partner(self.partner_id)
names = self._get_table_names(data) names = self._get_table_names(data)
tables = self.env['privacy.partner.data']
tables = self.env["privacy.partner.data"]
for name in sorted(names): for name in sorted(names):
vals = self._get_default_table( vals = self._get_default_table(
name=name,
data=[t for t in data if t[0] == name and not t[5]],
name=name, data=[t for t in data if t[0] == name and not t[5]],
) )
if vals: if vals:
tables |= self.env['privacy.partner.data'].create(vals)
tables |= self.env["privacy.partner.data"].create(vals)
self.table_ids = tables self.table_ids = tables
else: else:
self.table_ids = self.env['privacy.partner.data']
self.table_ids = self.env["privacy.partner.data"]
return { return {
'domain': {
'table_ids': [
('id', 'in', self.table_ids.ids)],
},
"domain": {"table_ids": [("id", "in", self.table_ids.ids)]},
} }
@api.onchange('company_id')
@api.onchange("company_id")
def _onchange_company_id(self): def _onchange_company_id(self):
if not self.company_id: if not self.company_id:
self.company_id = self.env.user.company_id self.company_id = self.env.user.company_id
return { return {
'domain': {
'partner_id': [
('company_id', 'in', [self.company_id.id, False])],
"domain": {
"partner_id": [("company_id", "in", [self.company_id.id, False])],
}, },
} }
def button_export_xlsx(self): def button_export_xlsx(self):
self.ensure_one() self.ensure_one()
if not self.table_ids: if not self.table_ids:
raise UserError(_('No data for this partner.'))
raise UserError(_("No data for this partner."))
return self.check_report(xlsx_report=True) return self.check_report(xlsx_report=True)
def _build_contexts(self, data): def _build_contexts(self, data):
result = {} result = {}
result['partner_id'] = data['form']['partner_id'][0] or False
result['company_id'] = data['form']['company_id'][0] or False
result['table_ids'] = 'table_ids' in data['form'] and \
data['form']['table_ids'] or False
result["partner_id"] = data["form"]["partner_id"][0] or False
result["company_id"] = data["form"]["company_id"][0] or False
result["table_ids"] = (
"table_ids" in data["form"] and data["form"]["table_ids"] or False
)
return result return result
@staticmethod @staticmethod
@ -86,54 +79,55 @@ class PrivacyPartnerReport(models.TransientModel):
for key, value in row.items(): for key, value in row.items():
label = self.env[model]._fields[key].string or key label = self.env[model]._fields[key].string or key
if self.env[model]._fields[key].store: if self.env[model]._fields[key].store:
if 'many2one' == self.env[model]._fields[key].type:
if "many2one" == self.env[model]._fields[key].type:
comodel = self.env[model]._fields[key].comodel_name comodel = self.env[model]._fields[key].comodel_name
if value: if value:
record = self.env[comodel].sudo().browse(value) record = self.env[comodel].sudo().browse(value)
cleaned_rows[i][label] = record.display_name cleaned_rows[i][label] = record.display_name
else: else:
cleaned_rows[i][label] = rows[i][key] cleaned_rows[i][label] = rows[i][key]
elif 'binary' == self.env[model]._fields[key].type:
elif "binary" == self.env[model]._fields[key].type:
binary = self._transform_binary(rows[i][key]) binary = self._transform_binary(rows[i][key])
if binary: if binary:
cleaned_rows[i][label] = binary cleaned_rows[i][label] = binary
elif '2many' not in self.env[model]._fields[key].type:
elif "2many" not in self.env[model]._fields[key].type:
cleaned_rows[i][label] = rows[i][key] cleaned_rows[i][label] = rows[i][key]
return cleaned_rows return cleaned_rows
def check_report(self, xlsx_report=False): def check_report(self, xlsx_report=False):
self.ensure_one() self.ensure_one()
data = {} data = {}
data['ids'] = self.env.context.get('active_ids', [])
data['model'] = self.env.context.get('active_model', 'ir.ui.menu')
data['form'] = self.read(['partner_id', 'company_id', 'table_ids'])[0]
data["ids"] = self.env.context.get("active_ids", [])
data["model"] = self.env.context.get("active_model", "ir.ui.menu")
data["form"] = self.read(["partner_id", "company_id", "table_ids"])[0]
used_context = self._build_contexts(data) used_context = self._build_contexts(data)
data['form']['id'] = str(data['form']['id'])
data['form']['used_context'] = dict(
used_context, lang=self.env.context.get('lang', 'en_US'))
data["form"]["id"] = str(data["form"]["id"])
data["form"]["used_context"] = dict(
used_context, lang=self.env.context.get("lang", "en_US")
)
return self._print_report(data=data, xlsx_report=xlsx_report) return self._print_report(data=data, xlsx_report=xlsx_report)
def compute_data_for_report(self, data): def compute_data_for_report(self, data):
if not data.get('form'):
if not data.get("form"):
raise UserError( raise UserError(
_("Form content is missing, this report cannot be printed."))
partner = data['form'].get('partner_id', False)
_("Form content is missing, this report cannot be printed.")
)
partner = data["form"].get("partner_id", False)
if not partner: if not partner:
raise UserError(
_("No provided partner."))
partner = self.env['res.partner'].sudo().browse(partner[0])
tables = data['form'].get('table_ids', False)
raise UserError(_("No provided partner."))
partner = self.env["res.partner"].sudo().browse(partner[0])
tables = data["form"].get("table_ids", False)
if tables: if tables:
tables = self.env['privacy.partner.data'].browse(tables)
tables = self.env["privacy.partner.data"].browse(tables)
tables = self._get_rows_from_tables(tables, partner) tables = self._get_rows_from_tables(tables, partner)
data.update({'tables': tables, })
data.update({"tables": tables})
return data return data
def _exclude_column(self, model, column): def _exclude_column(self, model, column):
# https://github.com/odoo/odoo/issues/24927 # https://github.com/odoo/odoo/issues/24927
if model in ('mail.compose.message', 'survey.mail.compose.message'):
if column in ('needaction_partner_ids', 'starred_partner_ids'):
if model in ("mail.compose.message", "survey.mail.compose.message"):
if column in ("needaction_partner_ids", "starred_partner_ids"):
return True return True
# feel free to add more specific cases meanwhile the issue is not fixed # feel free to add more specific cases meanwhile the issue is not fixed
@ -147,12 +141,14 @@ class PrivacyPartnerReport(models.TransientModel):
res |= self.env[t[1]].sudo().browse(t[3]) res |= self.env[t[1]].sudo().browse(t[3])
if res: if res:
values = { values = {
'name': name,
'model_id': self.env['ir.model'].sudo().search(
[('model', '=', res._name)]).id,
'count_rows': len(res.ids),
'field_type': field_type,
'res_ids': res.ids,
"name": name,
"model_id": self.env["ir.model"]
.sudo()
.search([("model", "=", res._name)])
.id,
"count_rows": len(res.ids),
"field_type": field_type,
"res_ids": res.ids,
} }
return values return values
return {} return {}
@ -166,12 +162,15 @@ class PrivacyPartnerReport(models.TransientModel):
def _get_rows_from_model(self, model, partner): def _get_rows_from_model(self, model, partner):
lines = self.env[model.model] lines = self.env[model.model]
columns = [k for k, v in self.env[model.model]._fields.items()
if v.comodel_name == 'res.partner' and
v.store and not self._exclude_column(model.model, k)]
columns = [
k
for k, v in self.env[model.model]._fields.items()
if v.comodel_name == "res.partner"
and v.store
and not self._exclude_column(model.model, k)
]
for column in columns: for column in columns:
lines |= self.env[model.model].sudo().search(
[(column, '=', partner.id)])
lines |= self.env[model.model].sudo().search([(column, "=", partner.id)])
rows = lines.sudo().read(load=False) rows = lines.sudo().read(load=False)
rows = self._clean_data(model.model, rows) rows = self._clean_data(model.model, rows)
return rows return rows
@ -191,15 +190,30 @@ class PrivacyPartnerReport(models.TransientModel):
return names return names
def _get_tables_from_partner(self, partner): def _get_tables_from_partner(self, partner):
tables = [t[0] for t in [
[[self.env[m]._table, m, k, self.env[m].sudo().search(
[(k, '=', partner.id)]).ids, v.type, self.env[m]._transient]
for k, v in self.env[m]._fields.items()
if v.comodel_name == 'res.partner' and self.env[m]._auto and
v.store and not self._exclude_column(m, k)]
for m in [x for x in self.env.registry.keys()]] if t]
tables = [
t[0]
for t in [
[
[
self.env[m]._table,
m,
k,
self.env[m].sudo().search([(k, "=", partner.id)]).ids,
v.type,
self.env[m]._transient,
]
for k, v in self.env[m]._fields.items()
if v.comodel_name == "res.partner"
and self.env[m]._auto
and v.store
and not self._exclude_column(m, k)
]
for m in [x for x in self.env.registry.keys()]
]
if t
]
for i, t in enumerate(tables): for i, t in enumerate(tables):
if t[4] == 'many2many':
if t[4] == "many2many":
if t[3]: if t[3]:
relation = self.env[t[1]]._fields[t[2]].relation relation = self.env[t[1]]._fields[t[2]].relation
if relation: if relation:
@ -207,44 +221,38 @@ class PrivacyPartnerReport(models.TransientModel):
return tables return tables
def _print_report(self, data, xlsx_report=False): def _print_report(self, data, xlsx_report=False):
records = self.env[data['model']].sudo().browse(data.get('ids', []))
records = self.env[data["model"]].sudo().browse(data.get("ids", []))
if xlsx_report: if xlsx_report:
return self.env.ref('privacy_partner_report.report_partner_xlsx').\
with_context(landscape=True).report_action(
records, data=data)
return (
self.env.ref("privacy_partner_report.report_partner_xlsx")
.with_context(landscape=True)
.report_action(records, data=data)
)
class PrivacyPartnerData(models.TransientModel): class PrivacyPartnerData(models.TransientModel):
_name = "privacy.partner.data" _name = "privacy.partner.data"
_description = "Privacy Partner Data" _description = "Privacy Partner Data"
name = fields.Char(
string='Database Table',
)
name = fields.Char(string="Database Table",)
model_id = fields.Many2one( model_id = fields.Many2one(
comodel_name='ir.model',
ondelete='cascade',
string='Models',
)
field_type = fields.Char(
string="Type", oldname='type',
comodel_name="ir.model", ondelete="cascade", string="Models",
) )
count_rows = fields.Integer(
default=0,
string='Number of lines',
field_type = fields.Char(string="Type", oldname="type",)
count_rows = fields.Integer(default=0, string="Number of lines",)
res_ids = fields.Char(
"Related Document IDs", index=True, help="List of Related Document IDs"
) )
res_ids = fields.Char('Related Document IDs', index=True,
help='List of Related Document IDs')
def action_view_records(self): def action_view_records(self):
self.ensure_one() self.ensure_one()
response = { response = {
'name': self.model_id.display_name,
'type': 'ir.actions.act_window',
'res_model': self.model_id.model,
'view_mode': 'tree,form',
'domain': [('id', 'in', ast.literal_eval(self.res_ids))],
'target': 'current',
'context': {'delete': True},
"name": self.model_id.display_name,
"type": "ir.actions.act_window",
"res_model": self.model_id.model,
"view_mode": "tree,form",
"domain": [("id", "in", ast.literal_eval(self.res_ids))],
"target": "current",
"context": {"delete": True},
} }
return response return response

84
privacy_partner_report/wizard/privacy_report_partner_wizard.xml

@ -1,49 +1,75 @@
<!-- Copyright 2018 Eficent Business and IT Consulting Services S.L. <!-- Copyright 2018 Eficent Business and IT Consulting Services S.L.
License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl-3.0) --> License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl-3.0) -->
<odoo> <odoo>
<record id="privacy_partner_report_wizard" model="ir.ui.view"> <record id="privacy_partner_report_wizard" model="ir.ui.view">
<field name="name">Partner Report</field> <field name="name">Partner Report</field>
<field name="model">privacy.partner.report</field> <field name="model">privacy.partner.report</field>
<field name="arch" type="xml"> <field name="arch" type="xml">
<form string="Report Options">
<header>
<button name="button_export_xlsx" string="Export XLSX" type="object" default_focus="1" class="oe_highlight"/>
</header>
<group groups="base.group_multi_company">
<group>
<span >Select a company</span><br/>
<field name="company_id" options="{'no_create': True}" nolabel="1" />
<form string="Report Options">
<header>
<button
name="button_export_xlsx"
string="Export XLSX"
type="object"
default_focus="1"
class="oe_highlight"
/>
</header>
<group groups="base.group_multi_company">
<group>
<span>Select a company</span>
<br />
<field
name="company_id"
options="{'no_create': True}"
nolabel="1"
/>
</group>
</group> </group>
</group>
<group>
<group> <group>
<span>Select a partner</span><br/>
<field name="partner_id" options="{'no_create': True}" nolabel="1" domain="[('company_id', 'in', [False, company_id])]"/>
<group>
<span>Select a partner</span>
<br />
<field
name="partner_id"
options="{'no_create': True}"
nolabel="1"
domain="[('company_id', 'in', [False, company_id])]"
/>
</group>
</group>
<group
string='Models with related partner data'
attrs="{'invisible': [('partner_id', '=', False)]}"
col="1"
>
<field
name="table_ids"
options="{'no_create': True}"
editable="bottom"
nolabel="1"
>
<tree>
<field name="model_id" />
<field name="count_rows" text-align="right" />
<button
name="action_view_records"
type="object"
icon="fa-info-circle"
attrs="{'invisible': [('count_rows', '=', '0')]}"
/>
</tree>
</field>
</group> </group>
</group>
<group string='Models with related partner data' attrs="{'invisible': [('partner_id', '=', False)]}" col="1">
<field name="table_ids" options="{'no_create': True}" editable="bottom" nolabel="1">
<tree>
<field name="model_id"/>
<field name="count_rows" text-align="right"/>
<button name="action_view_records"
type="object" icon="fa-info-circle"
attrs="{'invisible': [('count_rows', '=', '0')]}" />
</tree>
</field>
</group>
</form>
</form>
</field> </field>
</record> </record>
<record id="action_privacy_partner_menu" model="ir.actions.act_window"> <record id="action_privacy_partner_menu" model="ir.actions.act_window">
<field name="name">Partner Report</field> <field name="name">Partner Report</field>
<field name="res_model">privacy.partner.report</field> <field name="res_model">privacy.partner.report</field>
<field name="type">ir.actions.act_window</field> <field name="type">ir.actions.act_window</field>
<field name="view_mode">form</field> <field name="view_mode">form</field>
<field name="view_id" ref="privacy_partner_report_wizard"/>
<field name="view_id" ref="privacy_partner_report_wizard" />
<field name="target">current</field> <field name="target">current</field>
</record> </record>
</odoo> </odoo>
Loading…
Cancel
Save