diff --git a/contract_forecast/README.rst b/contract_forecast/README.rst new file mode 100644 index 00000000..b85d6cac --- /dev/null +++ b/contract_forecast/README.rst @@ -0,0 +1,74 @@ +================= +Contract Forecast +================= + +.. !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png + :target: https://odoo-community.org/page/development-status + :alt: Beta +.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png + :target: http://www.gnu.org/licenses/agpl-3.0-standalone.html + :alt: License: AGPL-3 +.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fcontract-lightgray.png?logo=github + :target: https://github.com/OCA/contract/tree/12.0/contract_forecast + :alt: OCA/contract +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/contract-12-0/contract-12-0-contract_forecast + :alt: Translate me on Weblate +.. |badge5| image:: https://img.shields.io/badge/runbot-Try%20me-875A7B.png + :target: https://runbot.odoo-community.org/runbot/110/12.0 + :alt: Try me on Runbot + +|badge1| |badge2| |badge3| |badge4| |badge5| + +This module add the possibility to analyse contract forecast. + +**Table of contents** + +.. contents:: + :local: + +Bug Tracker +=========== + +Bugs are tracked on `GitHub Issues `_. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us smashing it by providing a detailed and welcomed +`feedback `_. + +Do not contact contributors directly about support or help with technical issues. + +Credits +======= + +Authors +~~~~~~~ + +* ACSONE SA/NV + +Contributors +~~~~~~~~~~~~ + +* Souheil Bejaoui + + +Maintainers +~~~~~~~~~~~ + +This module is maintained by the OCA. + +.. image:: https://odoo-community.org/logo.png + :alt: Odoo Community Association + :target: https://odoo-community.org + +OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use. + +This module is part of the `OCA/contract `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/contract_forecast/__init__.py b/contract_forecast/__init__.py new file mode 100644 index 00000000..cc6b6354 --- /dev/null +++ b/contract_forecast/__init__.py @@ -0,0 +1,2 @@ +from . import models +from .hooks import post_init_hook diff --git a/contract_forecast/__manifest__.py b/contract_forecast/__manifest__.py new file mode 100644 index 00000000..01ebdf7d --- /dev/null +++ b/contract_forecast/__manifest__.py @@ -0,0 +1,20 @@ +# Copyright 2019 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +{ + "name": "Contract Forecast", + "summary": """ + Contract forecast""", + "version": "12.0.1.0.1", + "license": "AGPL-3", + "author": "ACSONE SA/NV," "Odoo Community Association (OCA)", + "website": "https://github.com/OCA/contract", + "depends": ["base", "contract", "queue_job"], + "data": [ + "security/contract_line_forecast_period.xml", + "views/contract_line_forecast_period.xml", + "views/contract.xml", + ], + "external_dependencies": {"python": ["dateutil"]}, + "post_init_hook": "post_init_hook", +} diff --git a/contract_forecast/hooks.py b/contract_forecast/hooks.py new file mode 100644 index 00000000..0faef878 --- /dev/null +++ b/contract_forecast/hooks.py @@ -0,0 +1,30 @@ +# Copyright 2019 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +import logging + +from odoo import api +from odoo.tools import SUPERUSER_ID + +_logger = logging.getLogger(__name__) + + +def post_init_hook(cr, registry): + """ + Generate contract line forecast periods + """ + _logger.info( + "Post init hook for module post_init_hook: " + "Generate contract line forecast periods" + ) + with api.Environment.manage(): + env = api.Environment(cr, SUPERUSER_ID, {}) + offset = 0 + while True: + contract_lines = env["contract.line"].search( + [('is_canceled', '=', False)], limit=100, offset=offset + ) + contract_lines.with_delay()._generate_forecast_periods() + if len(contract_lines) < 100: + break + offset += 100 diff --git a/contract_forecast/migrations/12.0.1.0.1/post-migration.py b/contract_forecast/migrations/12.0.1.0.1/post-migration.py new file mode 100644 index 00000000..487e537d --- /dev/null +++ b/contract_forecast/migrations/12.0.1.0.1/post-migration.py @@ -0,0 +1,18 @@ +# Copyright 2019 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +import logging + +_logger = logging.getLogger(__name__) + + +def migrate(cr, version): + """Set company_id for all forecasts""" + _logger.info("Set company_id for all forecasts") + cr.execute(""" + UPDATE contract_line_forecast_period AS forecast + SET company_id=contract.company_id + FROM contract_contract AS contract + WHERE forecast.contract_id=contract.id + AND forecast.contract_id IS NOT NULL + """) diff --git a/contract_forecast/models/__init__.py b/contract_forecast/models/__init__.py new file mode 100644 index 00000000..9fd5685c --- /dev/null +++ b/contract_forecast/models/__init__.py @@ -0,0 +1,4 @@ +from . import contract +from . import contract_line +from . import contract_line_forecast_period +from . import res_company diff --git a/contract_forecast/models/contract.py b/contract_forecast/models/contract.py new file mode 100644 index 00000000..f5c00138 --- /dev/null +++ b/contract_forecast/models/contract.py @@ -0,0 +1,42 @@ +# Copyright 2019 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import _, api, models + + +class ContractContract(models.Model): + + _inherit = "contract.contract" + + @api.multi + def action_show_contract_forecast(self): + self.ensure_one() + context = {'search_default_groupby_date_invoice': True} + context.update(self.env.context) + + return { + "type": "ir.actions.act_window", + "name": _("Contract Forecast"), + "res_model": "contract.line.forecast.period", + "domain": [("contract_id", "=", self.id)], + "view_mode": "pivot,tree", + "context": context, + } + + @api.model + def _get_forecast_update_trigger_fields(self): + return [] + + @api.multi + def write(self, values): + res = super(ContractContract, self).write(values) + if any( + [ + field in values + for field in self._get_forecast_update_trigger_fields() + ] + ): + for rec in self: + for contract_line in rec.contract_line_ids: + contract_line.with_delay()._generate_forecast_periods() + return res diff --git a/contract_forecast/models/contract_line.py b/contract_forecast/models/contract_line.py new file mode 100644 index 00000000..fcc98852 --- /dev/null +++ b/contract_forecast/models/contract_line.py @@ -0,0 +1,144 @@ +# Copyright 2019 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from dateutil.relativedelta import relativedelta +from odoo import api, fields, models +from odoo.addons.queue_job.job import job + +QUEUE_CHANNEL = "root.CONTRACT_FORECAST" + + +class ContractLine(models.Model): + + _inherit = "contract.line" + + forecast_period_ids = fields.One2many( + comodel_name="contract.line.forecast.period", + inverse_name="contract_line_id", + string="Forecast Periods", + required=False, + ) + + @api.multi + def _prepare_contract_line_forecast_period( + self, period_date_start, period_date_end, recurring_next_date + ): + self.ensure_one() + return { + "name": self._insert_markers(period_date_start, period_date_end), + "contract_id": self.contract_id.id, + "company_id": self.contract_id.company_id.id, + "contract_line_id": self.id, + "product_id": self.product_id.id, + "date_start": period_date_start, + "date_end": period_date_end, + "date_invoice": recurring_next_date, + "discount": self.discount, + "price_unit": self.price_unit, + "quantity": self._get_quantity_to_invoice( + period_date_start, period_date_end, recurring_next_date + ), + } + + @api.multi + def _get_contract_forecast_end_date(self): + self.ensure_one() + today = fields.Date.context_today(self) + return today + self.get_relative_delta( + self.contract_id.company_id.contract_forecast_rule_type, + self.contract_id.company_id.contract_forecast_interval, + ) + + @api.multi + def _get_generate_forecast_periods_criteria(self, period_date_end): + self.ensure_one() + if self.is_canceled or not self.active: + return False + contract_forecast_end_date = self._get_contract_forecast_end_date() + if not self.date_end or self.is_auto_renew: + return period_date_end < contract_forecast_end_date + return ( + period_date_end < self.date_end + and period_date_end < contract_forecast_end_date + ) + + @api.multi + @job(default_channel=QUEUE_CHANNEL) + def _generate_forecast_periods(self): + values = [] + for rec in self: + rec.forecast_period_ids.unlink() + if rec.recurring_next_date: + last_date_invoiced = ( + rec.last_date_invoiced + if rec.last_date_invoiced + else rec.date_start - relativedelta(days=1) + ) + period_date_end = last_date_invoiced + recurring_next_date = rec.recurring_next_date + while rec._get_generate_forecast_periods_criteria( + period_date_end + ): + period_dates = rec._get_period_to_invoice( + last_date_invoiced, + recurring_next_date, + stop_at_date_end=not rec.is_auto_renew, + ) + period_date_start, period_date_end, recurring_next_date = ( + period_dates + ) + values.append( + rec._prepare_contract_line_forecast_period( + period_date_start, + period_date_end, + recurring_next_date, + ) + ) + last_date_invoiced = period_date_end + recurring_next_date = ( + recurring_next_date + + self.get_relative_delta( + rec.recurring_rule_type, rec.recurring_interval + ) + ) + return self.env["contract.line.forecast.period"].create(values) + + @api.model + def create(self, values): + contract_lines = super(ContractLine, self).create(values) + for contract_line in contract_lines: + contract_line.with_delay()._generate_forecast_periods() + return contract_lines + + @api.model + def _get_forecast_update_trigger_fields(self): + return [ + "name", + "sequence", + "product_id", + "date_start", + "date_end", + "quantity", + "price_unit", + "discount", + "recurring_invoicing_type", + "recurring_next_date", + "recurring_rule_type", + "recurring_interval", + "is_canceled", + "active", + "is_auto_renew", + ] + + @api.multi + def write(self, values): + res = super(ContractLine, self).write(values) + if any( + [ + field in values + for field in self._get_forecast_update_trigger_fields() + ] + ): + for rec in self: + rec.with_delay()._generate_forecast_periods() + return res diff --git a/contract_forecast/models/contract_line_forecast_period.py b/contract_forecast/models/contract_line_forecast_period.py new file mode 100644 index 00000000..158b85b6 --- /dev/null +++ b/contract_forecast/models/contract_line_forecast_period.py @@ -0,0 +1,77 @@ +# Copyright 2019 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import api, fields, models +from odoo.addons import decimal_precision as dp + + +class ContractLineForecastPeriod(models.Model): + + _name = "contract.line.forecast.period" + _description = "Contract Line Forecast Period" + _order = "date_invoice, sequence" + + name = fields.Char(string="Name", required=True, readonly=True) + sequence = fields.Integer( + string="Sequence", related="contract_line_id.sequence", store=True + ) + contract_id = fields.Many2one( + comodel_name="contract.contract", + string="Contract", + required=True, + readonly=True, + ondelete="cascade", + related="contract_line_id.contract_id", + store=True, + index=True, + ) + contract_line_id = fields.Many2one( + comodel_name="contract.line", + string="Contract Line", + required=True, + readonly=True, + ondelete="cascade", + index=True, + ) + product_id = fields.Many2one( + comodel_name="product.product", + string="Product", + required=True, + readonly=True, + related="contract_line_id.product_id", + store=True, + index=True, + ) + date_start = fields.Date(string="Date Start", required=True, readonly=True) + date_end = fields.Date(string="Date End", required=True, readonly=True) + date_invoice = fields.Date( + string="Invoice Date", required=True, readonly=True + ) + quantity = fields.Float(default=1.0, required=True) + price_unit = fields.Float(string='Unit Price') + price_subtotal = fields.Float( + digits=dp.get_precision("Account"), + string="Amount Untaxed", + compute='_compute_price_subtotal', + store=True + ) + discount = fields.Float( + string='Discount (%)', + digits=dp.get_precision('Discount'), + help='Discount that is applied in generated invoices.' + ' It should be less or equal to 100', + ) + company_id = fields.Many2one(comodel_name="res.company", string="Company") + + @api.multi + @api.depends('quantity', 'price_unit', 'discount') + def _compute_price_subtotal(self): + for line in self: + subtotal = line.quantity * line.price_unit + discount = line.discount / 100 + subtotal *= 1 - discount + if line.contract_id.pricelist_id: + cur = line.contract_id.pricelist_id.currency_id + line.price_subtotal = cur.round(subtotal) + else: + line.price_subtotal = subtotal diff --git a/contract_forecast/models/res_company.py b/contract_forecast/models/res_company.py new file mode 100644 index 00000000..1adbadf6 --- /dev/null +++ b/contract_forecast/models/res_company.py @@ -0,0 +1,16 @@ +# Copyright 2019 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import fields, models + + +class ResCompany(models.Model): + + _inherit = "res.company" + + contract_forecast_interval = fields.Integer( + string="Number of contract forecast Periods", default=12 + ) + contract_forecast_rule_type = fields.Selection( + [("monthly", "Month(s)"), ("yearly", "Year(s)")], default="monthly" + ) diff --git a/contract_forecast/readme/CONTRIBUTORS.rst b/contract_forecast/readme/CONTRIBUTORS.rst new file mode 100644 index 00000000..16f45059 --- /dev/null +++ b/contract_forecast/readme/CONTRIBUTORS.rst @@ -0,0 +1,2 @@ +* Souheil Bejaoui + diff --git a/contract_forecast/readme/DESCRIPTION.rst b/contract_forecast/readme/DESCRIPTION.rst new file mode 100644 index 00000000..42a43a93 --- /dev/null +++ b/contract_forecast/readme/DESCRIPTION.rst @@ -0,0 +1 @@ +This module add the possibility to analyse contract forecast. diff --git a/contract_forecast/security/contract_line_forecast_period.xml b/contract_forecast/security/contract_line_forecast_period.xml new file mode 100644 index 00000000..f58b3042 --- /dev/null +++ b/contract_forecast/security/contract_line_forecast_period.xml @@ -0,0 +1,25 @@ + + + + + + + contract.line.forecast.period user access + + + + + + + + + + Forecast multi company rule + + + ['|',('company_id','=',False),('company_id','child_of',[user.company_id.id])] + + + + diff --git a/contract_forecast/static/description/index.html b/contract_forecast/static/description/index.html new file mode 100644 index 00000000..2336055c --- /dev/null +++ b/contract_forecast/static/description/index.html @@ -0,0 +1,419 @@ + + + + + + +Contract Forecast + + + +
+

Contract Forecast

+ + +

Beta License: AGPL-3 OCA/contract Translate me on Weblate Try me on Runbot

+

This module add the possibility to analyse contract forecast.

+

Table of contents

+ +
+

Bug Tracker

+

Bugs are tracked on GitHub Issues. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us smashing it by providing a detailed and welcomed +feedback.

+

Do not contact contributors directly about support or help with technical issues.

+
+
+

Credits

+
+

Authors

+
    +
  • ACSONE SA/NV
  • +
+
+
+

Contributors

+ +
+
+

Maintainers

+

This module is maintained by the OCA.

+Odoo Community Association +

OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use.

+

This module is part of the OCA/contract project on GitHub.

+

You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.

+
+
+
+ + diff --git a/contract_forecast/tests/__init__.py b/contract_forecast/tests/__init__.py new file mode 100644 index 00000000..60071c1e --- /dev/null +++ b/contract_forecast/tests/__init__.py @@ -0,0 +1 @@ +from . import test_contract_line_forecast_period diff --git a/contract_forecast/tests/test_contract_line_forecast_period.py b/contract_forecast/tests/test_contract_line_forecast_period.py new file mode 100644 index 00000000..702c169b --- /dev/null +++ b/contract_forecast/tests/test_contract_line_forecast_period.py @@ -0,0 +1,160 @@ +# Copyright 2019 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from datetime import date +from dateutil.relativedelta import relativedelta + +from odoo.addons.contract.tests.test_contract import TestContractBase +from odoo.tools import mute_logger + + +class TestContractLineForecastPeriod(TestContractBase): + @mute_logger("odoo.addons.queue_job.models.base") + def setUp(self): + self.env = self.env( + context=dict(self.env.context, test_queue_job_no_delay=True) + ) + super(TestContractLineForecastPeriod, self).setUp() + self.this_year = date.today().year + self.line_vals["date_start"] = date.today() + self.line_vals["recurring_next_date"] = date.today() + self.acct_line = self.env["contract.line"].create(self.line_vals) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_creation(self): + self.acct_line.write( + { + 'date_start': "{this_year}-01-01".format( + this_year=self.this_year), + 'recurring_next_date': "{this_year}-01-01".format( + this_year=self.this_year), + 'date_end': "{this_year}-12-31".format( + this_year=self.this_year), + 'recurring_rule_type': "monthly", + 'recurring_invoicing_type': 'pre-paid', + } + ) + self.assertTrue(self.acct_line.forecast_period_ids) + self.assertEqual(len(self.acct_line.forecast_period_ids), 12) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_1(self): + self.acct_line.write( + { + 'date_start': "{this_year}-01-01".format( + this_year=self.this_year), + 'recurring_next_date': "{this_year}-01-01".format( + this_year=self.this_year), + 'date_end': "{this_year}-12-31".format( + this_year=self.this_year), + 'recurring_rule_type': "yearly", + 'recurring_invoicing_type': 'pre-paid', + } + ) + self.assertTrue(self.acct_line.forecast_period_ids) + self.assertEqual(len(self.acct_line.forecast_period_ids), 1) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_2(self): + self.acct_line.write( + { + 'date_start': "{this_year}-01-01".format( + this_year=self.this_year), + 'recurring_next_date': "{this_year}-01-31".format( + this_year=self.this_year), + 'date_end': "{this_year}-06-05".format( + this_year=self.this_year), + 'recurring_rule_type': "monthlylastday", + 'recurring_invoicing_type': 'pre-paid', + } + ) + self.assertTrue(self.acct_line.forecast_period_ids) + self.assertEqual(len(self.acct_line.forecast_period_ids), 6) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_3(self): + self.assertEqual(self.acct_line.price_subtotal, 50) + self.acct_line.write({"price_unit": 50}) + self.assertEqual(self.acct_line.price_subtotal, 25) + self.assertEqual( + self.acct_line.forecast_period_ids[0].price_subtotal, 25 + ) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_4(self): + self.assertEqual(self.acct_line.price_subtotal, 50) + self.acct_line.write({"discount": 0}) + self.assertEqual(self.acct_line.price_subtotal, 100) + self.assertEqual( + self.acct_line.forecast_period_ids[0].price_subtotal, 100 + ) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_5(self): + self.acct_line.cancel() + self.assertFalse(self.acct_line.forecast_period_ids) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_6(self): + self.acct_line.write( + { + 'date_start': "{this_year}-01-01".format( + this_year=self.this_year), + 'recurring_next_date': "{this_year}-01-01".format( + this_year=self.this_year), + 'date_end': "{this_year}-01-28".format( + this_year=self.this_year), + 'recurring_rule_type': "monthly", + 'recurring_invoicing_type': 'pre-paid', + } + ) + self.assertTrue(self.acct_line.forecast_period_ids) + self.assertEqual(len(self.acct_line.forecast_period_ids), 1) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_6(self): + self.acct_line.write( + { + 'date_start': "{this_year}-01-01".format( + this_year=self.this_year), + 'recurring_next_date': "{this_year}-02-01".format( + this_year=self.this_year), + 'date_end': "{this_year}-01-28".format( + this_year=self.this_year), + 'recurring_rule_type': "monthly", + 'recurring_invoicing_type': 'post-paid', + } + ) + self.assertTrue(self.acct_line.forecast_period_ids) + self.assertEqual(len(self.acct_line.forecast_period_ids), 1) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_7(self): + self.acct_line.write( + { + 'date_end': date.today() + relativedelta(months=3), + 'recurring_rule_type': "monthlylastday", + 'recurring_invoicing_type': 'pre-paid', + 'is_auto_renew': True, + } + ) + self.acct_line._onchange_date_start() + self.assertTrue(self.acct_line.forecast_period_ids) + self.assertEqual(len(self.acct_line.forecast_period_ids), 13) + + @mute_logger("odoo.addons.queue_job.models.base") + def test_forecast_period_on_contract_line_update_8(self): + self.acct_line.write( + { + 'date_start': "{this_year}-01-14".format( + this_year=self.this_year), + 'recurring_next_date': "{this_year}-01-31".format( + this_year=self.this_year), + 'date_end': "{this_year}-01-14".format( + this_year=self.this_year), + 'recurring_rule_type': "monthlylastday", + 'recurring_invoicing_type': 'post-paid', + } + ) + self.assertTrue(self.acct_line.forecast_period_ids) + self.assertEqual(len(self.acct_line.forecast_period_ids), 1) diff --git a/contract_forecast/views/contract.xml b/contract_forecast/views/contract.xml new file mode 100644 index 00000000..387288ad --- /dev/null +++ b/contract_forecast/views/contract.xml @@ -0,0 +1,28 @@ + + + + + + + contract.contract.form (in contract_forcast) + contract.contract + + + + + + + + + + diff --git a/contract_forecast/views/contract_line_forecast_period.xml b/contract_forecast/views/contract_line_forecast_period.xml new file mode 100644 index 00000000..2f155916 --- /dev/null +++ b/contract_forecast/views/contract_line_forecast_period.xml @@ -0,0 +1,60 @@ + + + + + + + + contract.line.forecast.period.search (in + contract_forecast) + + contract.line.forecast.period + + + + + + + + + + + + + + + contract.line.forecast.period.tree (in + contract_forecast) + + contract.line.forecast.period + + + + + + + + + + + + + + contract.line.forecast.period.tree (in + contract_forecast) + + contract.line.forecast.period + + + + + + + + + + diff --git a/oca_dependencies.txt b/oca_dependencies.txt index 02b6e1e6..d7f9e901 100644 --- a/oca_dependencies.txt +++ b/oca_dependencies.txt @@ -1,2 +1,3 @@ bank-payment +queue diff --git a/setup/contract_forecast/odoo/addons/contract_forecast b/setup/contract_forecast/odoo/addons/contract_forecast new file mode 120000 index 00000000..f79c4fb1 --- /dev/null +++ b/setup/contract_forecast/odoo/addons/contract_forecast @@ -0,0 +1 @@ +../../../../contract_forecast \ No newline at end of file diff --git a/setup/contract_forecast/setup.py b/setup/contract_forecast/setup.py new file mode 100644 index 00000000..28c57bb6 --- /dev/null +++ b/setup/contract_forecast/setup.py @@ -0,0 +1,6 @@ +import setuptools + +setuptools.setup( + setup_requires=['setuptools-odoo'], + odoo_addon=True, +)