You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
123 lines
4.1 KiB
123 lines
4.1 KiB
# -*- coding: utf-8 -*-
|
|
# © 2014-2016 Therp BV <http://therp.nl>
|
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
|
from odoo import _, api, models, fields
|
|
from odoo.exceptions import UserError
|
|
from odoo.addons.base.ir.ir_model import MODULE_UNINSTALL_FLAG
|
|
|
|
|
|
class IrModel(models.Model):
|
|
_inherit = 'ir.model'
|
|
|
|
def _drop_table(self):
|
|
# Allow to skip this step during model unlink
|
|
# The super method crashes if the model cannot be instantiated
|
|
if self.env.context.get('no_drop_table'):
|
|
return True
|
|
return super(IrModel, self)._drop_table()
|
|
|
|
@api.depends()
|
|
def _inherited_models(self):
|
|
"""this function crashes for undefined models"""
|
|
existing_model_ids = self.filtered(lambda x: x.model in self.env)
|
|
super(IrModel, existing_model_ids)._inherited_models()
|
|
|
|
|
|
class IrModelFields(models.Model):
|
|
_inherit = 'ir.model.fields'
|
|
|
|
@api.multi
|
|
def _prepare_update(self):
|
|
# Allow to skip this step during model unlink
|
|
# The super method crashes if the model cannot be instantiated
|
|
if self.env.context.get('no_prepare_update'):
|
|
return True
|
|
return super(IrModelFields, self)._prepare_update()
|
|
|
|
|
|
class CleanupPurgeLineModel(models.TransientModel):
|
|
_inherit = 'cleanup.purge.line'
|
|
_name = 'cleanup.purge.line.model'
|
|
_description = 'Purge models'
|
|
|
|
wizard_id = fields.Many2one(
|
|
'cleanup.purge.wizard.model', 'Purge Wizard', readonly=True)
|
|
|
|
@api.multi
|
|
def purge(self):
|
|
"""
|
|
Unlink models upon manual confirmation.
|
|
"""
|
|
context_flags = {
|
|
MODULE_UNINSTALL_FLAG: True,
|
|
'no_drop_table': True,
|
|
'purge': True,
|
|
'no_prepare_update': True,
|
|
}
|
|
|
|
if self:
|
|
objs = self
|
|
else:
|
|
objs = self.env['cleanup.purge.line.model']\
|
|
.browse(self._context.get('active_ids'))
|
|
for line in objs:
|
|
self.env.cr.execute(
|
|
"SELECT id, model from ir_model WHERE model = %s",
|
|
(line.name,))
|
|
row = self.env.cr.fetchone()
|
|
if not row:
|
|
continue
|
|
self.logger.info('Purging model %s', row[1])
|
|
attachments = self.env['ir.attachment'].search([
|
|
('res_model', '=', line.name)
|
|
])
|
|
if attachments:
|
|
self.env.cr.execute(
|
|
"UPDATE ir_attachment SET res_model = NULL "
|
|
"WHERE id in %s",
|
|
(tuple(attachments.ids), ))
|
|
self.env['ir.model.constraint'].search([
|
|
('model', '=', line.name),
|
|
]).unlink()
|
|
relations = self.env['ir.model.fields'].search([
|
|
('relation', '=', row[1]),
|
|
]).with_context(**context_flags)
|
|
for relation in relations:
|
|
try:
|
|
# Fails if the model on the target side
|
|
# cannot be instantiated
|
|
relation.unlink()
|
|
except KeyError:
|
|
pass
|
|
except AttributeError:
|
|
pass
|
|
self.env['ir.model.relation'].search([
|
|
('model', '=', line.name)
|
|
]).with_context(**context_flags).unlink()
|
|
self.env['ir.model'].browse([row[0]])\
|
|
.with_context(**context_flags).unlink()
|
|
line.write({'purged': True})
|
|
return True
|
|
|
|
|
|
class CleanupPurgeWizardModel(models.TransientModel):
|
|
_inherit = 'cleanup.purge.wizard'
|
|
_name = 'cleanup.purge.wizard.model'
|
|
_description = 'Purge models'
|
|
|
|
@api.model
|
|
def find(self):
|
|
"""
|
|
Search for models that cannot be instantiated.
|
|
"""
|
|
res = []
|
|
self.env.cr.execute("SELECT model from ir_model")
|
|
for model, in self.env.cr.fetchall():
|
|
if model not in self.env:
|
|
res.append((0, 0, {'name': model}))
|
|
if not res:
|
|
raise UserError(_('No orphaned models found'))
|
|
return res
|
|
|
|
purge_line_ids = fields.One2many(
|
|
'cleanup.purge.line.model', 'wizard_id', 'Models to purge')
|