Compare commits
34 Commits
Author | SHA1 | Date |
---|---|---|
Fabien BOURGEOIS | 8220716619 | |
Fabien BOURGEOIS | 1532369619 | |
Fabien BOURGEOIS | 08c6493de6 | |
Fabien BOURGEOIS | b236af81f2 | |
Fabien BOURGEOIS | 0c51c735b0 | |
Fabien BOURGEOIS | 64ea17bae0 | |
Fabien BOURGEOIS | 937df77a72 | |
Fabien BOURGEOIS | b6f62571b2 | |
Fabien BOURGEOIS | f5d90d83cf | |
Fabien BOURGEOIS | 8767b1aaeb | |
Fabien BOURGEOIS | 401b3b5ffc | |
Fabien BOURGEOIS | d7aeafdc6f | |
Fabien BOURGEOIS | 265453589b | |
Fabien BOURGEOIS | 2e501018d8 | |
Fabien BOURGEOIS | 0d40afd498 | |
Fabien BOURGEOIS | 360832d876 | |
Fabien BOURGEOIS | 264d1e37b2 | |
Fabien BOURGEOIS | c338738e2b | |
Fabien BOURGEOIS | 4f6399c576 | |
Fabien BOURGEOIS | ebef43ad8a | |
Fabien BOURGEOIS | d0a7b6a74e | |
Fabien BOURGEOIS | f6b892efd4 | |
Fabien BOURGEOIS | f2c8818e43 | |
Fabien BOURGEOIS | aa7bfe9449 | |
Fabien BOURGEOIS | 291d4fc394 | |
Fabien BOURGEOIS | 095f4ce7d9 | |
Fabien BOURGEOIS | dd846361cf | |
Fabien BOURGEOIS | d46a51a330 | |
Fabien BOURGEOIS | 2a1a765222 | |
Fabien BOURGEOIS | 3b3846526e | |
Fabien BOURGEOIS | b7c203ac0d | |
Fabien BOURGEOIS | 88e66e0fde | |
Fabien BOURGEOIS | 08d143a5f6 | |
Fabien BOURGEOIS | 4ca4d5169c |
|
@ -22,12 +22,12 @@
|
||||||
* For CSV published by Crédit Coopératif on their website customer space ;
|
* For CSV published by Crédit Coopératif on their website customer space ;
|
||||||
* Adds option on bank import statement wizard ;
|
* Adds option on bank import statement wizard ;
|
||||||
* Checks and processes file.""",
|
* Checks and processes file.""",
|
||||||
'version': '10.0.0.1.3',
|
'version': '12.0.0.1.3',
|
||||||
'category': 'Banking addons',
|
'category': 'Banking addons',
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'application': False,
|
'application': False,
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['account_bank_statement_import'],
|
'depends': ['account_bank_statement_import'],
|
||||||
'data': ['wizard/account_bank_statement_import_views.xml']
|
'data': ['wizard/account_bank_statement_import_views.xml']
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2022 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from . import override
|
|
@ -0,0 +1,37 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2022 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
{
|
||||||
|
'name': 'DBFilter from configuration',
|
||||||
|
'summary': 'Database filtering from configuration',
|
||||||
|
'description': """ Database filtering from configuration :
|
||||||
|
- use Odoo own filtering mechanism fist ;
|
||||||
|
- then apply configuration : `domains_to_databases` : string in form of `database:domain1,domain2|database2:domain3`.
|
||||||
|
|
||||||
|
Needs to be added to `server_wide_modules` on configuration or `--load`,
|
||||||
|
for example `server_wide_modules = base,web,dbfilter_from_config`.
|
||||||
|
Also required : `proxy_mode` to `True`
|
||||||
|
|
||||||
|
Inspired from OCA `dbfilter_from_header`.""",
|
||||||
|
'version': '12.0.0.0.6',
|
||||||
|
'category': 'Yaltik',
|
||||||
|
'author': 'Fabien Bourgeois',
|
||||||
|
'license': 'AGPL-3',
|
||||||
|
'application': False,
|
||||||
|
'installable': True,
|
||||||
|
'depends': ['base']
|
||||||
|
}
|
|
@ -0,0 +1,64 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2022 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
""" Override of databse filtering to allow new features """
|
||||||
|
|
||||||
|
from re import match
|
||||||
|
from json import loads
|
||||||
|
from odoo import http
|
||||||
|
from odoo.tools import config
|
||||||
|
|
||||||
|
db_filter_odoo = http.db_filter
|
||||||
|
|
||||||
|
# Cache usage
|
||||||
|
db_filter_dict = {}
|
||||||
|
all_hosts = []
|
||||||
|
db_filter_str = config.get('database_from_hosts', False)
|
||||||
|
if db_filter_str:
|
||||||
|
for entry in db_filter_str.split('|'):
|
||||||
|
database, hosts = entry.split(':')
|
||||||
|
hosts = hosts.split(',')
|
||||||
|
db_filter_dict[database] = hosts
|
||||||
|
all_hosts += hosts
|
||||||
|
http_host_resolv = {}
|
||||||
|
|
||||||
|
def db_filter(dbs, httprequest=None):
|
||||||
|
""" Override db_filter """
|
||||||
|
if db_filter_str:
|
||||||
|
httprequest = httprequest or http.request.httprequest
|
||||||
|
http_host = httprequest.environ.get('HTTP_HOST', '').split(':')[0]
|
||||||
|
# If in cache and not empty
|
||||||
|
if http_host in http_host_resolv and http_host_resolv.get(http_host):
|
||||||
|
dbs = http_host_resolv[http_host]
|
||||||
|
elif http_host in all_hosts:
|
||||||
|
for database, hosts in db_filter_dict.items():
|
||||||
|
for host in hosts:
|
||||||
|
if host == http_host:
|
||||||
|
dbs = [i for i in dbs if match(database, i)]
|
||||||
|
break
|
||||||
|
if http_host not in http_host_resolv:
|
||||||
|
http_host_resolv[http_host] = dbs
|
||||||
|
else:
|
||||||
|
dbs = db_filter_odoo(dbs, httprequest)
|
||||||
|
else:
|
||||||
|
dbs = db_filter_odoo(dbs, httprequest)
|
||||||
|
return dbs
|
||||||
|
|
||||||
|
|
||||||
|
if (config.get('proxy_mode')
|
||||||
|
and 'dbfilter_from_config' in config.get('server_wide_modules')):
|
||||||
|
http.db_filter = db_filter
|
|
@ -20,12 +20,12 @@
|
||||||
'name': 'Grants follow-ups',
|
'name': 'Grants follow-ups',
|
||||||
'summary': 'Grants follow-ups ',
|
'summary': 'Grants follow-ups ',
|
||||||
'description': 'Grants follow-ups ',
|
'description': 'Grants follow-ups ',
|
||||||
'version': '10.0.1.0.0',
|
'version': '12.0.1.0.0',
|
||||||
'category': 'Membership',
|
'category': 'Membership',
|
||||||
'author': 'Fabien Bourgeois, Youssef ELOUAHBY',
|
'author': 'Fabien Bourgeois, Youssef ELOUAHBY',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'application': True,
|
'application': True,
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['mail', 'document'],
|
'depends': ['mail', 'document'],
|
||||||
'data': ['security/security.xml',
|
'data': ['security/security.xml',
|
||||||
'security/ir.model.access.csv',
|
'security/ir.model.access.csv',
|
||||||
|
|
|
@ -24,7 +24,7 @@
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'application': False,
|
'application': False,
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['note'],
|
'depends': ['note'],
|
||||||
'data': ['views/note_view.xml']
|
'data': ['views/note_view.xml']
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'application': False,
|
'application': False,
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['note_base'],
|
'depends': ['note_base'],
|
||||||
'data': ['views/note_dates.xml']
|
'data': ['views/note_dates.xml']
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,7 +24,7 @@
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'application': False,
|
'application': False,
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['note_base'],
|
'depends': ['note_base'],
|
||||||
'data': ['views/note_priority.xml']
|
'data': ['views/note_priority.xml']
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,7 @@
|
||||||
'category': 'Tools',
|
'category': 'Tools',
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['note'],
|
'depends': ['note'],
|
||||||
'data': ['note_tags_view.xml']
|
'data': ['note_tags_view.xml']
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,7 @@
|
||||||
'category': 'Tools',
|
'category': 'Tools',
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['note_base', 'project'],
|
'depends': ['note_base', 'project'],
|
||||||
'data': ['note_task_view.xml']
|
'data': ['note_task_view.xml']
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,7 +34,7 @@
|
||||||
<t t-esc="o.name">
|
<t t-esc="o.name">
|
||||||
</t>
|
</t>
|
||||||
""",
|
""",
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'application': True,
|
'application': True,
|
||||||
'data':[
|
'data':[
|
||||||
'view/templates.xml',
|
'view/templates.xml',
|
||||||
|
|
|
@ -18,11 +18,11 @@
|
||||||
{
|
{
|
||||||
'name': 'Email widget check syntax',
|
'name': 'Email widget check syntax',
|
||||||
'summary': 'Extends Web Email widget to ensure valid syntax',
|
'summary': 'Extends Web Email widget to ensure valid syntax',
|
||||||
'version': '10.0.1.0.1',
|
'version': '12.0.1.0.1',
|
||||||
'category': 'Usability',
|
'category': 'Usability',
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': [ 'web' ],
|
'depends': [ 'web' ],
|
||||||
'data': [ 'view.xml' ]
|
'data': [ 'view.xml' ]
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,11 +18,11 @@
|
||||||
{
|
{
|
||||||
'name': 'French format phone widget check',
|
'name': 'French format phone widget check',
|
||||||
'summary': 'Extends Web Char widget to handle french format phone',
|
'summary': 'Extends Web Char widget to handle french format phone',
|
||||||
'version': '10.0.1.0.2',
|
'version': '12.0.1.0.2',
|
||||||
'category': 'Usability',
|
'category': 'Usability',
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': [ 'web' ],
|
'depends': [ 'web' ],
|
||||||
'data': [ 'view.xml' ],
|
'data': [ 'view.xml' ],
|
||||||
'qweb': ['static/src/xml/qweb.xml']
|
'qweb': ['static/src/xml/qweb.xml']
|
||||||
|
|
|
@ -18,11 +18,11 @@
|
||||||
{
|
{
|
||||||
'name': 'URL widget check syntax',
|
'name': 'URL widget check syntax',
|
||||||
'summary': 'Extends Web URL widget to ensure valid syntax',
|
'summary': 'Extends Web URL widget to ensure valid syntax',
|
||||||
'version': '10.0.1.0.1',
|
'version': '12.0.1.0.1',
|
||||||
'category': 'Usability',
|
'category': 'Usability',
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': [ 'web' ],
|
'depends': [ 'web' ],
|
||||||
'data': [ 'view.xml' ]
|
'data': [ 'view.xml' ]
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2019 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from . import odoo
|
|
@ -0,0 +1,29 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2019 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
{
|
||||||
|
'name': 'Odoo XML DSL base module and fns',
|
||||||
|
'summary': 'Odoo XML Domain Specific Language base module and functions',
|
||||||
|
'description': """ Odoo XML Domain Specific Language base module and functions """,
|
||||||
|
'version': '12.0.0.1.0',
|
||||||
|
'category': 'Yaltik',
|
||||||
|
'author': 'Fabien Bourgeois',
|
||||||
|
'license': 'AGPL-3',
|
||||||
|
'application': False,
|
||||||
|
'installable': True,
|
||||||
|
'depends': ['base']
|
||||||
|
}
|
|
@ -0,0 +1,46 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright 2019 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
""" XML helpers and macros """
|
||||||
|
|
||||||
|
from xml.etree import ElementTree as ET
|
||||||
|
|
||||||
|
def xmlroot(tree):
|
||||||
|
""" Special process for root XML Node """
|
||||||
|
rootel = ET.Element(tree['tag'], tree['attrs'])
|
||||||
|
children = tree['children']
|
||||||
|
if children:
|
||||||
|
xmlchild(rootel, children)
|
||||||
|
return rootel
|
||||||
|
|
||||||
|
def xmlchild(parent, children):
|
||||||
|
""" Handling of children (ie non root) XML Nodes with/o text and
|
||||||
|
subchildren (recursive) """
|
||||||
|
for child in children:
|
||||||
|
if isinstance(child, str):
|
||||||
|
parent.text = child
|
||||||
|
else:
|
||||||
|
attrs = {unicode(k): unicode(v) for k, v in child['attrs'].items()}
|
||||||
|
new_parent = ET.SubElement(parent, child['tag'], attrs)
|
||||||
|
subchildren = child['children']
|
||||||
|
if subchildren:
|
||||||
|
xmlchild(new_parent, subchildren)
|
||||||
|
|
||||||
|
def xmln(tag='', attrs=None, children=None, text=False):
|
||||||
|
""" XMLNode with default children, not attributes """
|
||||||
|
children = ([text] if text else children) or []
|
||||||
|
return {'tag': tag, 'attrs': attrs or {}, 'children': children}
|
|
@ -0,0 +1,81 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright 2019 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
""" Odoo XML DSL """
|
||||||
|
|
||||||
|
from . import base as x
|
||||||
|
|
||||||
|
def odoo(*args):
|
||||||
|
""" odoo tag shortcut """
|
||||||
|
return x.xmlroot(x.xmln('odoo', {}, *args))
|
||||||
|
def data(*args):
|
||||||
|
""" data tag shortcut """
|
||||||
|
return x.xmln('data', {}, *args)
|
||||||
|
def record(*args):
|
||||||
|
""" record tag shortcut """
|
||||||
|
return x.xmln('record', *args)
|
||||||
|
|
||||||
|
def view(xmlid, children):
|
||||||
|
""" view tag shortcut """
|
||||||
|
return record({'id': xmlid, 'model': 'ir.ui.view'}, children)
|
||||||
|
|
||||||
|
def view_def(xmlid, name, model, arch):
|
||||||
|
""" View and first fields simplification with record xmlid, name, targeted model """
|
||||||
|
return view(xmlid, [field_name(name), field_model(model), field_arch(arch)])
|
||||||
|
|
||||||
|
def view_inherit(name, model, inherit, arch):
|
||||||
|
""" Inherited View simplification with name of the record, xmlid for model
|
||||||
|
and inherited view """
|
||||||
|
module = __name__.split('.')[2]
|
||||||
|
inherited = inherit.split('.')[1]
|
||||||
|
xmlid = '%s_inherit_%s' % (inherited, module)
|
||||||
|
return view(xmlid, [field_name(name), field_model(model),
|
||||||
|
field_inherit(inherit), field_arch(arch)])
|
||||||
|
|
||||||
|
def actions_server_code(xmlid, name, modelref, code):
|
||||||
|
""" Server actions of type code """
|
||||||
|
return record({'id': xmlid, 'model': 'ir.actions.server'},
|
||||||
|
[field_name(name),
|
||||||
|
field({'name': 'model_id', 'ref': modelref}),
|
||||||
|
field({'name': 'state'}, ['code']),
|
||||||
|
field({'name': 'code'}, [code])])
|
||||||
|
|
||||||
|
def field(*args):
|
||||||
|
""" field tag shortcut """
|
||||||
|
return x.xmln('field', *args)
|
||||||
|
def field_name(name):
|
||||||
|
""" field name tag shortcut """
|
||||||
|
return field({'name': 'name'}, [name])
|
||||||
|
def field_model(model):
|
||||||
|
""" field model tag shortcut """
|
||||||
|
return field({'name': 'model'}, [model])
|
||||||
|
def field_inherit(xmlid):
|
||||||
|
""" field inherit tag shortcut """
|
||||||
|
return field({'name': 'inherit_id', 'ref': xmlid})
|
||||||
|
def field_arch(*args):
|
||||||
|
""" field arch tag shortcut """
|
||||||
|
return field({'name': 'arch', 'type': 'xml'}, *args)
|
||||||
|
|
||||||
|
def xml_write(mpath, filename, tree):
|
||||||
|
""" Write XML file according to filename and given tree """
|
||||||
|
import os.path
|
||||||
|
from xml.etree import ElementTree as ET
|
||||||
|
output_xml = ET.tostring(tree).decode('utf-8')
|
||||||
|
output_path = os.path.dirname(os.path.abspath(mpath))
|
||||||
|
fpath = u'%s/%s' % (output_path, filename)
|
||||||
|
with open(fpath, 'w') as xml_file:
|
||||||
|
xml_file.write(output_xml)
|
|
@ -25,7 +25,7 @@
|
||||||
'description': """ Yaltik Backend Green theme, based upon great
|
'description': """ Yaltik Backend Green theme, based upon great
|
||||||
DodgerBlue Backend Theme from OpenWorx :
|
DodgerBlue Backend Theme from OpenWorx :
|
||||||
https://github.com/Openworx/themes/tree/8.0/dodgerblue_backend_theme """,
|
https://github.com/Openworx/themes/tree/8.0/dodgerblue_backend_theme """,
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['base'],
|
'depends': ['base'],
|
||||||
'data': ['views.xml']
|
'data': ['views.xml']
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2018 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
@ -0,0 +1,32 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2018 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
{
|
||||||
|
'name': 'Yaltik bootstrap',
|
||||||
|
'summary': 'Yaltik bootstrap : common dependencies for all projects',
|
||||||
|
'description': """ Yaltik bootstrap : common dependencies for all projects :
|
||||||
|
* addons from OCA : FR localization, debranding ;
|
||||||
|
* new backend and responsive theme """,
|
||||||
|
'version': '12.0.0.1.0',
|
||||||
|
'category': 'Yaltik',
|
||||||
|
'author': 'Fabien Bourgeois',
|
||||||
|
'license': 'AGPL-3',
|
||||||
|
'application': False,
|
||||||
|
'installable': True,
|
||||||
|
'depends': ['l10n_fr', 'l10n_fr_siret', 'l10n_fr_state', 'l10n_fr_department',
|
||||||
|
'disable_odoo_online', 'mail_debrand', 'muk_web_theme']
|
||||||
|
}
|
|
@ -18,11 +18,11 @@
|
||||||
{
|
{
|
||||||
'name': 'CRM Actions',
|
'name': 'CRM Actions',
|
||||||
'summary': 'Action management, instead of new activity, in CRM',
|
'summary': 'Action management, instead of new activity, in CRM',
|
||||||
'version': '10.0.1.3.5',
|
'version': '12.0.1.3.5',
|
||||||
'category': 'Sales',
|
'category': 'Sales',
|
||||||
'author': 'Fabien BOURGEOIS - Yaltik',
|
'author': 'Fabien BOURGEOIS - Yaltik',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'application': True,
|
'application': True,
|
||||||
'depends': ['crm', 'calendar'],
|
'depends': ['crm', 'calendar'],
|
||||||
'data': ['views/crm_action_views.xml',
|
'data': ['views/crm_action_views.xml',
|
||||||
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
.PHONY: clean
|
||||||
|
clean:
|
||||||
|
find . -name '*.pyc' -delete
|
||||||
|
find . -name '__pycache__' -delete
|
||||||
|
|
||||||
|
.PHONY: test
|
||||||
|
test:
|
||||||
|
python3 tests/test_xml_base.py
|
||||||
|
python3 tests/test_odoo.py
|
|
@ -0,0 +1,18 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2019-2020 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from . import src
|
|
@ -0,0 +1,27 @@
|
||||||
|
# Copyright 2019-2024 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
{
|
||||||
|
'name': 'Yaltik Odoo DSL base module and fns',
|
||||||
|
'summary': 'Yaltik Odoo Domain Specific Language base module and functions',
|
||||||
|
'description': """ Yaltik Odoo Domain Specific Language base module and functions """,
|
||||||
|
'version': '12.0.0.5.13',
|
||||||
|
'category': 'Yaltik',
|
||||||
|
'author': 'Fabien Bourgeois',
|
||||||
|
'license': 'AGPL-3',
|
||||||
|
'application': False,
|
||||||
|
'installable': True,
|
||||||
|
'depends': ['base']
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright 2020 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from . import odoo_dsl
|
|
@ -0,0 +1,144 @@
|
||||||
|
# Copyright 2019-2024 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
""" Odoo XML DSL """
|
||||||
|
|
||||||
|
from .xml_base import xmlroot, xmln
|
||||||
|
|
||||||
|
# XML helpers functions and macros
|
||||||
|
|
||||||
|
# Odoo root XML Node
|
||||||
|
odoo = lambda children: xmlroot({'tag': 'odoo', 'attrs': {}, 'children': children})
|
||||||
|
|
||||||
|
# Special data Node
|
||||||
|
def data(*args):
|
||||||
|
""" Allow optional args on data tag """
|
||||||
|
if len(args) == 1:
|
||||||
|
args = list(args)
|
||||||
|
args.insert(0, {})
|
||||||
|
return xmln('data', *args)
|
||||||
|
|
||||||
|
# Aliases
|
||||||
|
function = lambda *args: xmln('function', *args)
|
||||||
|
record = lambda *args: xmln('record', *args)
|
||||||
|
form = lambda *args: xmln('form', *args)
|
||||||
|
tree = lambda *args: xmln('tree', *args)
|
||||||
|
search = lambda *args: xmln('search', *args)
|
||||||
|
template = lambda *args: xmln('template', *args)
|
||||||
|
|
||||||
|
# Actions
|
||||||
|
act_window = lambda *args: xmln('act_window', *args)
|
||||||
|
def act_window_model(model, attrs):
|
||||||
|
""" Build new act_window from model and args """
|
||||||
|
xmlid = '%s_view_action' % (model.replace('.', '_'))
|
||||||
|
name = '%s Action' % ' '.join(map(lambda w: w.capitalize(), model.split('.')))
|
||||||
|
attrs_clone = attrs.copy() # Avoid side-effect
|
||||||
|
attrs_clone.update({'id': xmlid, 'name': name, 'res_model': model})
|
||||||
|
return act_window(attrs_clone)
|
||||||
|
|
||||||
|
def action_server_code(xmlid, name, modelref, code):
|
||||||
|
""" Server actions of type code """
|
||||||
|
children = [field_name(name),
|
||||||
|
field({'name': 'model_id', 'ref': modelref}, []),
|
||||||
|
field({'name': 'state'}, ['code']),
|
||||||
|
field({'name': 'code'}, [code])]
|
||||||
|
return record({'id': xmlid, 'model': 'ir.actions.server'}, children)
|
||||||
|
|
||||||
|
def client_action_multi(xmlid, name, model, action):
|
||||||
|
""" Client action multi (ir.values), with own xmlid, name, targeted model
|
||||||
|
and action """
|
||||||
|
action = "'ir.actions.server,%d'%{}".format(action)
|
||||||
|
children = [field_name(name),
|
||||||
|
field({'name': 'key2', 'eval': "'client_action_multi'"}),
|
||||||
|
field({'name': 'model', 'eval': "'%s'" % model}),
|
||||||
|
field({'name': 'value', 'eval': action})]
|
||||||
|
return record({'id': xmlid, 'model': 'ir.values'}, children)
|
||||||
|
|
||||||
|
# Menus
|
||||||
|
menuitem = lambda *args: xmln('menuitem', *args)
|
||||||
|
def menuitem_model(model, attrs):
|
||||||
|
""" Build new menuitem from model and attrs """
|
||||||
|
model_und = model.replace('.', '_')
|
||||||
|
xmlid = '%s_menu' % model_und
|
||||||
|
actionid = '%s_view_action' % model_und
|
||||||
|
attrs_clone = attrs.copy() # Avoid side-effect
|
||||||
|
attrs_clone.update({'id': xmlid, 'action': actionid})
|
||||||
|
return menuitem(attrs_clone)
|
||||||
|
|
||||||
|
# Form aliases
|
||||||
|
group = lambda *args: xmln('group', *args)
|
||||||
|
header = lambda *args: xmln('header', *args)
|
||||||
|
footer = lambda *args: xmln('footer', *args)
|
||||||
|
sheet = lambda *args: xmln('sheet', *args)
|
||||||
|
button = lambda *args: xmln('button', *args)
|
||||||
|
p = lambda *args: xmln('p', *args)
|
||||||
|
i = lambda *args: xmln('i', *args)
|
||||||
|
t = lambda *args: xmln('t', *args)
|
||||||
|
div = lambda *args: xmln('div', *args)
|
||||||
|
span = lambda *args: xmln('span', *args)
|
||||||
|
br = lambda *args: xmln('br', *args)
|
||||||
|
h1 = lambda *args: xmln('h1', *args)
|
||||||
|
h2 = lambda *args: xmln('h1', *args)
|
||||||
|
h3 = lambda *args: xmln('h1', *args)
|
||||||
|
h4 = lambda *args: xmln('h1', *args)
|
||||||
|
h5 = lambda *args: xmln('h1', *args)
|
||||||
|
ul = lambda *args: xmln('ul', *args)
|
||||||
|
li = lambda *args: xmln('li', *args)
|
||||||
|
label = lambda *args: xmln('label', *args)
|
||||||
|
strong = lambda *args: xmln('strong', *args)
|
||||||
|
small = lambda *args: xmln('small', *args)
|
||||||
|
table = lambda *args: xmln('table', *args)
|
||||||
|
tr = lambda *args: xmln('tr', *args)
|
||||||
|
td = lambda *args: xmln('td', *args)
|
||||||
|
notebook = lambda *args: xmln('notebook', *args)
|
||||||
|
page = lambda *args: xmln('page', *args)
|
||||||
|
xpath = lambda *args: xmln('xpath', *args)
|
||||||
|
attribute = lambda name, value: xmln('attribute', {'name': name}, [value])
|
||||||
|
|
||||||
|
# Fields
|
||||||
|
field = lambda *args: xmln('field', *args)
|
||||||
|
field_name = lambda name: field({'name': 'name'}, [name])
|
||||||
|
field_model = lambda model: field({'name': 'model'}, [model])
|
||||||
|
field_inherit = lambda xmlid: field({'name': 'inherit_id', 'ref': xmlid}, [])
|
||||||
|
field_arch = lambda *args: field({'name': 'arch', 'type': 'xml'}, *args)
|
||||||
|
|
||||||
|
# Search
|
||||||
|
filter = lambda *args: xmln('filter', *args)
|
||||||
|
|
||||||
|
# Views
|
||||||
|
view = lambda xmlid, children: record({'id': xmlid, 'model': 'ir.ui.view'}, children)
|
||||||
|
|
||||||
|
def view_def(xmlid, name, model, arch):
|
||||||
|
""" Shortcut for new view """
|
||||||
|
return view(xmlid, [field_name(name), field_model(model), field_arch(arch)])
|
||||||
|
|
||||||
|
def view_new(view_type, model, arch):
|
||||||
|
""" View : new view definition, based on type (form, tree, ...) and model ID """
|
||||||
|
model_und = model.replace('.', '_')
|
||||||
|
model_cap = ' '.join(map(lambda w: w.capitalize(), model.split('.')))
|
||||||
|
xmlid = "%s_view_%s" % (model_und, view_type)
|
||||||
|
name = ' '.join([model_cap, view_type.capitalize()])
|
||||||
|
return view_def(xmlid, name, model, arch)
|
||||||
|
|
||||||
|
def view_inherit(filename, model, inherit, arch):
|
||||||
|
""" Inherited View simplification with name of the record, xmlid for model
|
||||||
|
and inherited view """
|
||||||
|
module = filename.split('.')[2]
|
||||||
|
inherited = inherit.split('.')[1]
|
||||||
|
xmlid = '%s_inherit_%s' % (inherited, module)
|
||||||
|
model_cap = ' '.join(map(lambda w: w.capitalize(), model.split('.')))
|
||||||
|
name = '%s Adaptations' % model_cap
|
||||||
|
return view(xmlid, [field_name(name), field_model(model),
|
||||||
|
field_inherit(inherit), field_arch(arch)])
|
|
@ -0,0 +1,75 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright 2019-2020 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
""" XML helpers and macros """
|
||||||
|
|
||||||
|
from os import path
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from xml.dom import minidom
|
||||||
|
from collections import namedtuple
|
||||||
|
from functools import partial
|
||||||
|
|
||||||
|
XMLDictElement = namedtuple('XMLDictElement', ['tag', 'attrs', 'children'])
|
||||||
|
|
||||||
|
|
||||||
|
def xmlroot(tree):
|
||||||
|
""" Special process for root XML Node """
|
||||||
|
rootel = ET.Element(tree['tag'], tree['attrs'])
|
||||||
|
if 'children' in tree:
|
||||||
|
xmlchild(rootel, tree['children'])
|
||||||
|
return rootel
|
||||||
|
|
||||||
|
def xmlchild(parent, children):
|
||||||
|
""" Handling of children (ie non root) XML Nodes with/o text and
|
||||||
|
subchildren (recursive) """
|
||||||
|
if isinstance(children, str):
|
||||||
|
parent.text = children
|
||||||
|
elif isinstance(children, XMLDictElement):
|
||||||
|
attrs = {str(k): str(v) for [k, v] in children.attrs.items()}
|
||||||
|
new_parent = ET.SubElement(parent, children.tag, attrs)
|
||||||
|
subchildren = children.children
|
||||||
|
if subchildren:
|
||||||
|
xmlchild(new_parent, subchildren)
|
||||||
|
elif isinstance(children, list):
|
||||||
|
list(map(partial(xmlchild, parent), children))
|
||||||
|
else:
|
||||||
|
raise TypeError('Invalid arguments for xmlchild')
|
||||||
|
|
||||||
|
def xmln(tag='', attrs={}, children=[]):
|
||||||
|
""" XMLDictElement building from dict object, with defaults """
|
||||||
|
if isinstance(attrs, list):
|
||||||
|
children = attrs
|
||||||
|
attrs = {}
|
||||||
|
xmldictel = partial(XMLDictElement, tag, attrs)
|
||||||
|
if isinstance(children, str):
|
||||||
|
return xmldictel([children])
|
||||||
|
if isinstance(children, list):
|
||||||
|
return xmldictel(children)
|
||||||
|
raise TypeError('Invalid arguments for xmln')
|
||||||
|
|
||||||
|
|
||||||
|
def xml_write(filepath, tree, pretty=True):
|
||||||
|
""" Write XML file according to filename and given tree """
|
||||||
|
if filepath.endswith('.py'): # if .pyc, no need to generate XML
|
||||||
|
output_xml = ET.tostring(tree)
|
||||||
|
if pretty:
|
||||||
|
output_xml = minidom.parseString(output_xml).toprettyxml(indent=' ')
|
||||||
|
output_path = path.abspath(filepath).split(u'/')
|
||||||
|
output_path[-1] = output_path[-1].replace(u'.py', u'_views.xml')
|
||||||
|
output_path = u'/'.join(output_path)
|
||||||
|
with open(output_path, 'w') as output_file:
|
||||||
|
output_file.write(output_xml)
|
|
@ -0,0 +1,247 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright 2020 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
""" Odoo Helpers tests """
|
||||||
|
|
||||||
|
import unittest
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from src.xml_base import XMLDictElement
|
||||||
|
import src.odoo_dsl as od
|
||||||
|
|
||||||
|
class TestOdooBase(unittest.TestCase):
|
||||||
|
""" Odoo Helpers tests """
|
||||||
|
|
||||||
|
def test_odoo(self):
|
||||||
|
""" Test odoo function """
|
||||||
|
element = od.odoo([])
|
||||||
|
self.assertIsInstance(element, ET.Element)
|
||||||
|
self.assertEqual(element.tag, 'odoo')
|
||||||
|
|
||||||
|
def test_data(self):
|
||||||
|
""" Test data function """
|
||||||
|
element = od.data([])
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'data')
|
||||||
|
self.assertEqual(element.attrs, {})
|
||||||
|
|
||||||
|
element = od.data({"one": "attribute"}, [])
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'data')
|
||||||
|
self.assertEqual(element.attrs, {"one": "attribute"})
|
||||||
|
|
||||||
|
def test_aliases(self):
|
||||||
|
""" Test simple aliases to xmln """
|
||||||
|
element = od.record({"one": "attribute"}, 'A child')
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'record')
|
||||||
|
self.assertEqual(element.attrs, {"one": "attribute"})
|
||||||
|
self.assertEqual(element.children, ['A child'])
|
||||||
|
|
||||||
|
element = od.tree()
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'tree')
|
||||||
|
|
||||||
|
def test_act_window_model(self):
|
||||||
|
""" Test act_window function """
|
||||||
|
element = od.act_window_model('sample.model', {'view_type': 'form'})
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'act_window')
|
||||||
|
self.assertEqual(element.attrs['view_type'], 'form')
|
||||||
|
self.assertEqual(element.attrs['id'], 'sample_model_view_action')
|
||||||
|
self.assertEqual(element.attrs['res_model'], 'sample.model')
|
||||||
|
self.assertEqual(element.attrs['name'], 'Sample Model Action')
|
||||||
|
|
||||||
|
def test_menunitem_model(self):
|
||||||
|
""" Test menuitem function """
|
||||||
|
element = od.menuitem_model('sample.model', {'groups': 'base.user_employee'})
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'menuitem')
|
||||||
|
self.assertEqual(element.attrs['groups'], 'base.user_employee')
|
||||||
|
self.assertEqual(element.attrs['id'], 'sample_model_menu')
|
||||||
|
self.assertEqual(element.attrs['action'], 'sample_model_view_action')
|
||||||
|
|
||||||
|
def test_attribute(self):
|
||||||
|
""" Test attribute function """
|
||||||
|
element = od.attribute('invisible', "1")
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'attribute')
|
||||||
|
self.assertEqual(element.attrs['name'], 'invisible')
|
||||||
|
self.assertEqual(element.children, ["1"])
|
||||||
|
|
||||||
|
def test_fields(self):
|
||||||
|
""" Test fields function """
|
||||||
|
element = od.field({"one": "attribute"}, 'A child')
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'field')
|
||||||
|
|
||||||
|
element = od.field_name('A name')
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'field')
|
||||||
|
self.assertEqual(element.attrs['name'], 'name')
|
||||||
|
self.assertEqual(element.children, ['A name'])
|
||||||
|
|
||||||
|
element = od.field_model('sample.model')
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'field')
|
||||||
|
self.assertEqual(element.attrs['name'], 'model')
|
||||||
|
self.assertEqual(element.children, ['sample.model'])
|
||||||
|
|
||||||
|
element = od.field_inherit('module.xml_view')
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'field')
|
||||||
|
self.assertEqual(element.attrs['name'], 'inherit_id')
|
||||||
|
self.assertEqual(element.attrs['ref'], 'module.xml_view')
|
||||||
|
self.assertFalse(element.children)
|
||||||
|
|
||||||
|
element = od.field_arch()
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'field')
|
||||||
|
self.assertEqual(element.attrs['name'], 'arch')
|
||||||
|
self.assertEqual(element.attrs['type'], 'xml')
|
||||||
|
self.assertFalse(element.children)
|
||||||
|
|
||||||
|
def test_view(self):
|
||||||
|
""" Test view function """
|
||||||
|
element = od.view('view_xmlid', [])
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'record')
|
||||||
|
self.assertEqual(element.attrs['id'], 'view_xmlid')
|
||||||
|
self.assertEqual(element.attrs['model'], 'ir.ui.view')
|
||||||
|
self.assertFalse(element.children)
|
||||||
|
|
||||||
|
def test_view_def(self):
|
||||||
|
""" Test view_def function """
|
||||||
|
element = od.view_def('view_xmlid', 'View', 'sample.model', [])
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'record')
|
||||||
|
self.assertEqual(element.attrs['id'], 'view_xmlid')
|
||||||
|
self.assertEqual(element.attrs['model'], 'ir.ui.view')
|
||||||
|
self.assertEqual((len(element.children)), 3)
|
||||||
|
self.assertIsInstance(element.children[0], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[0].tag, 'field')
|
||||||
|
self.assertEqual(element.children[0].attrs['name'], 'name')
|
||||||
|
self.assertEqual(element.children[0].children, ['View'])
|
||||||
|
self.assertIsInstance(element.children[1], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[1].tag, 'field')
|
||||||
|
self.assertEqual(element.children[1].attrs['name'], 'model')
|
||||||
|
self.assertEqual(element.children[1].children, ['sample.model'])
|
||||||
|
self.assertIsInstance(element.children[2], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[2].tag, 'field')
|
||||||
|
self.assertEqual(element.children[2].attrs['name'], 'arch')
|
||||||
|
self.assertEqual(element.children[2].attrs['type'], 'xml')
|
||||||
|
self.assertFalse(element.children[2].children)
|
||||||
|
|
||||||
|
def test_view_new(self):
|
||||||
|
""" Test view_new function """
|
||||||
|
element = od.view_new('tree', 'sample.model', [])
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'record')
|
||||||
|
self.assertEqual(element.attrs['id'], 'sample_model_view_tree')
|
||||||
|
self.assertEqual(element.attrs['model'], 'ir.ui.view')
|
||||||
|
self.assertEqual((len(element.children)), 3)
|
||||||
|
self.assertIsInstance(element.children[0], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[0].tag, 'field')
|
||||||
|
self.assertEqual(element.children[0].attrs['name'], 'name')
|
||||||
|
self.assertEqual(element.children[0].children, ['Sample Model Tree'])
|
||||||
|
self.assertIsInstance(element.children[1], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[1].tag, 'field')
|
||||||
|
self.assertEqual(element.children[1].attrs['name'], 'model')
|
||||||
|
self.assertEqual(element.children[1].children, ['sample.model'])
|
||||||
|
self.assertIsInstance(element.children[2], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[2].tag, 'field')
|
||||||
|
self.assertEqual(element.children[2].attrs['name'], 'arch')
|
||||||
|
self.assertEqual(element.children[2].attrs['type'], 'xml')
|
||||||
|
self.assertFalse(element.children[2].children)
|
||||||
|
|
||||||
|
def test_view_inherit(self):
|
||||||
|
""" Test view_inherit function """
|
||||||
|
element = od.view_inherit('odoo.addons.module', 'sample.model', 'parent.view', [])
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'record')
|
||||||
|
self.assertEqual(element.attrs['id'], 'view_inherit_module')
|
||||||
|
self.assertEqual(element.attrs['model'], 'ir.ui.view')
|
||||||
|
self.assertEqual((len(element.children)), 4)
|
||||||
|
self.assertIsInstance(element.children[0], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[0].tag, 'field')
|
||||||
|
self.assertEqual(element.children[0].attrs['name'], 'name')
|
||||||
|
self.assertEqual(element.children[0].children, ['Sample Model Adaptations'])
|
||||||
|
self.assertIsInstance(element.children[1], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[1].tag, 'field')
|
||||||
|
self.assertEqual(element.children[1].attrs['name'], 'model')
|
||||||
|
self.assertEqual(element.children[1].children, ['sample.model'])
|
||||||
|
self.assertIsInstance(element.children[2], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[2].tag, 'field')
|
||||||
|
self.assertEqual(element.children[2].attrs['name'], 'inherit_id')
|
||||||
|
self.assertFalse(element.children[2].children)
|
||||||
|
self.assertIsInstance(element.children[3], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[3].tag, 'field')
|
||||||
|
self.assertEqual(element.children[3].attrs['name'], 'arch')
|
||||||
|
self.assertEqual(element.children[3].attrs['type'], 'xml')
|
||||||
|
self.assertFalse(element.children[3].children)
|
||||||
|
|
||||||
|
def test_action_server_code(self):
|
||||||
|
""" Test action_server_code function """
|
||||||
|
element = od.action_server_code('sample.xmlid', 'Code', 'sample.model',
|
||||||
|
'''record.do_something()''')
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'record')
|
||||||
|
self.assertEqual(element.attrs['id'], 'sample.xmlid')
|
||||||
|
self.assertEqual(element.attrs['model'], 'ir.actions.server')
|
||||||
|
self.assertEqual((len(element.children)), 4)
|
||||||
|
self.assertIsInstance(element.children[0], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[0].tag, 'field')
|
||||||
|
self.assertEqual(element.children[0].attrs['name'], 'name')
|
||||||
|
self.assertEqual(element.children[0].children, ['Code'])
|
||||||
|
self.assertIsInstance(element.children[1], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[1].tag, 'field')
|
||||||
|
self.assertEqual(element.children[1].attrs['name'], 'model_id')
|
||||||
|
self.assertEqual(element.children[1].attrs['ref'], 'sample.model')
|
||||||
|
self.assertFalse(element.children[1].children)
|
||||||
|
self.assertEqual(element.children[2].tag, 'field')
|
||||||
|
self.assertEqual(element.children[2].attrs['name'], 'state')
|
||||||
|
self.assertEqual(element.children[2].children, ['code'])
|
||||||
|
self.assertEqual(element.children[3].tag, 'field')
|
||||||
|
self.assertEqual(element.children[3].attrs['name'], 'code')
|
||||||
|
self.assertEqual(element.children[3].children, ['record.do_something()'])
|
||||||
|
|
||||||
|
def test_client_action_multi(self):
|
||||||
|
""" Test client_action_multi function """
|
||||||
|
element = od.client_action_multi('sample.xmlid', 'Multi', 'sample.model', 'sample.action')
|
||||||
|
self.assertIsInstance(element, XMLDictElement)
|
||||||
|
self.assertEqual(element.tag, 'record')
|
||||||
|
self.assertEqual(element.attrs['id'], 'sample.xmlid')
|
||||||
|
self.assertEqual(element.attrs['model'], 'ir.values')
|
||||||
|
self.assertEqual((len(element.children)), 4)
|
||||||
|
self.assertIsInstance(element.children[0], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[0].tag, 'field')
|
||||||
|
self.assertEqual(element.children[0].attrs['name'], 'name')
|
||||||
|
self.assertEqual(element.children[0].children, ['Multi'])
|
||||||
|
self.assertIsInstance(element.children[1], XMLDictElement)
|
||||||
|
self.assertEqual(element.children[1].tag, 'field')
|
||||||
|
self.assertEqual(element.children[1].attrs['name'], 'key2')
|
||||||
|
self.assertEqual(element.children[1].attrs['eval'], "'client_action_multi'")
|
||||||
|
self.assertFalse(element.children[1].children)
|
||||||
|
self.assertEqual(element.children[2].tag, 'field')
|
||||||
|
self.assertEqual(element.children[2].attrs['name'], 'model')
|
||||||
|
self.assertEqual(element.children[2].attrs['eval'], "'sample.model'")
|
||||||
|
self.assertEqual(element.children[3].tag, 'field')
|
||||||
|
self.assertEqual(element.children[3].attrs['name'], 'value')
|
||||||
|
self.assertEqual(element.children[3].attrs['eval'], "'ir.actions.server,%d'%sample.action")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main()
|
|
@ -0,0 +1,120 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright 2020 Fabien Bourgeois <fabien@yaltik.com>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
""" XML Helpers tests """
|
||||||
|
|
||||||
|
from functools import partial
|
||||||
|
import unittest
|
||||||
|
import xml.etree.ElementTree as ET
|
||||||
|
from os import unlink
|
||||||
|
from src.xml_base import xmln, xmlroot, xmlchild, xml_write
|
||||||
|
|
||||||
|
|
||||||
|
class TestXMLBase(unittest.TestCase):
|
||||||
|
""" XML Helpers tests """
|
||||||
|
|
||||||
|
def test_xmln(self):
|
||||||
|
""" Text xmln """
|
||||||
|
# Tags
|
||||||
|
self.assertDictEqual(xmln()._asdict(), {'tag': '', 'attrs': {}, 'children': []})
|
||||||
|
self.assertEqual(xmln('a tag').tag, 'a tag')
|
||||||
|
# Attrs
|
||||||
|
self.assertDictEqual(xmln(attrs={'a good': 'one'}).attrs, {'a good': 'one'})
|
||||||
|
# Childrens
|
||||||
|
self.assertListEqual(xmln(children=[1, 2, 3]).children, [1, 2, 3])
|
||||||
|
self.assertListEqual(xmln(children='Some text').children, ['Some text'])
|
||||||
|
|
||||||
|
with self.assertRaisesRegex(TypeError, 'Invalid arguments'):
|
||||||
|
xmln(children=False)
|
||||||
|
|
||||||
|
# Ensure that only children after tags is managed
|
||||||
|
element = xmln('tag', {'something': 'inside'})
|
||||||
|
self.assertIsInstance(element.attrs, dict)
|
||||||
|
self.assertIsInstance(element.children, list)
|
||||||
|
|
||||||
|
element = xmln('tag', ['something', 'inside'])
|
||||||
|
self.assertIsInstance(element.attrs, dict)
|
||||||
|
self.assertIsInstance(element.children, list)
|
||||||
|
|
||||||
|
|
||||||
|
def test_xmlchild(self):
|
||||||
|
""" Test xmlchild """
|
||||||
|
parent = xmlroot({'tag': 'root', 'attrs': {}, 'children': []})
|
||||||
|
xmlc_par = partial(xmlchild, parent)
|
||||||
|
|
||||||
|
# Bad arguments
|
||||||
|
with self.assertRaisesRegex(TypeError, 'Invalid arguments for xmlchild'):
|
||||||
|
xmlc_par(False)
|
||||||
|
# Need XMLDictElement, not dict
|
||||||
|
with self.assertRaisesRegex(TypeError, 'Invalid arguments for xmlchild'):
|
||||||
|
xmlc_par([{'tag': 't', 'attrs': {'a': 'b'}, 'children': []}])
|
||||||
|
|
||||||
|
xmlc_par(['some text'])
|
||||||
|
self.assertEqual(parent.text, 'some text')
|
||||||
|
|
||||||
|
xmlc_par([xmln('t', {'a': 'b'}, [])])
|
||||||
|
child = parent.iter('t').__next__()
|
||||||
|
self.assertEqual(child.tag, 't')
|
||||||
|
self.assertDictEqual(child.attrib, {'a': 'b'})
|
||||||
|
self.assertListEqual(list(child), [])
|
||||||
|
|
||||||
|
xmlc_par([xmln('t2', {1: 2}, [])])
|
||||||
|
child = parent.iter('t2').__next__()
|
||||||
|
self.assertDictEqual(child.attrib, {'1': '2'})
|
||||||
|
|
||||||
|
xmlc_par([xmln('tchildren', {}, [xmln('subchild', {}, [])])])
|
||||||
|
child = parent.iter('tchildren').__next__()
|
||||||
|
subchildren = list(child)
|
||||||
|
self.assertEqual(len(subchildren), 1)
|
||||||
|
self.assertEqual(subchildren[0].tag, 'subchild')
|
||||||
|
|
||||||
|
|
||||||
|
def test_xmlroot(self):
|
||||||
|
""" Test xmlroot """
|
||||||
|
root = xmlroot({'tag': 'root', 'attrs': {}, 'children': []})
|
||||||
|
self.assertIsInstance(root, ET.Element)
|
||||||
|
|
||||||
|
with self.assertRaisesRegex(TypeError, 'not subscriptable'):
|
||||||
|
xmlroot(False)
|
||||||
|
with self.assertRaisesRegex(KeyError, 'tag'):
|
||||||
|
xmlroot({})
|
||||||
|
with self.assertRaisesRegex(KeyError, 'attrs'):
|
||||||
|
xmlroot({'tag': 'root'})
|
||||||
|
|
||||||
|
|
||||||
|
def test_xml_write(self):
|
||||||
|
""" test xml_write """
|
||||||
|
children = [xmln('child1', {'attr': 'value'}, []),
|
||||||
|
xmln('child2', {}, "Some text")]
|
||||||
|
tree = xmlroot({'tag': 'root', 'attrs': {}, 'children': children})
|
||||||
|
xmlw = lambda p: xml_write(p, tree)
|
||||||
|
|
||||||
|
self.assertIsNone(xmlw('/badpath'))
|
||||||
|
self.assertIsNone(xmlw('/bad.ext'))
|
||||||
|
|
||||||
|
xmlw(__file__)
|
||||||
|
filepath = __file__.replace('.py', '_views.xml')
|
||||||
|
with open(filepath, 'r') as output_file:
|
||||||
|
output_xml = output_file.read()
|
||||||
|
self.assertIn('<?xml version', output_xml)
|
||||||
|
self.assertIn('<root>', output_xml)
|
||||||
|
self.assertIn('<child1 attr="value"/>', output_xml)
|
||||||
|
self.assertIn('<child2>Some text</child2>', output_xml)
|
||||||
|
unlink(filepath)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main()
|
|
@ -25,7 +25,7 @@
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'application': False,
|
'application': False,
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['portal_project'],
|
'depends': ['portal_project'],
|
||||||
'data': [
|
'data': [
|
||||||
'security/ir.model.access.csv',
|
'security/ir.model.access.csv',
|
||||||
|
|
|
@ -24,7 +24,7 @@
|
||||||
'author': 'Fabien Bourgeois',
|
'author': 'Fabien Bourgeois',
|
||||||
'license': 'AGPL-3',
|
'license': 'AGPL-3',
|
||||||
'application': False,
|
'application': False,
|
||||||
'installable': True,
|
'installable': False,
|
||||||
'depends': ['portal_project_issue'],
|
'depends': ['portal_project_issue'],
|
||||||
'data': [
|
'data': [
|
||||||
'security/ir.model.access.csv',
|
'security/ir.model.access.csv',
|
||||||
|
|
Loading…
Reference in New Issue