diff --git a/event_rest_api/README.rst b/event_rest_api/README.rst new file mode 100644 index 000000000..1b68b411d --- /dev/null +++ b/event_rest_api/README.rst @@ -0,0 +1,78 @@ +============== +Event Rest Api +============== + +.. !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! 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%2Fevent-lightgray.png?logo=github + :target: https://github.com/OCA/event/tree/14.0/event_rest_api + :alt: OCA/event +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/event-14-0/event-14-0-event_rest_api + :alt: Translate me on Weblate +.. |badge5| image:: https://img.shields.io/badge/runbot-Try%20me-875A7B.png + :target: https://runbot.odoo-community.org/runbot/199/14.0 + :alt: Try me on Runbot + +|badge1| |badge2| |badge3| |badge4| |badge5| + +TODO + +**Table of contents** + +.. contents:: + :local: + +Usage +===== + +TODO + +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 +~~~~~~~~~~~~ + +* Quentin Groulard + +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/event `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/event_rest_api/__init__.py b/event_rest_api/__init__.py new file mode 100644 index 000000000..882bbab34 --- /dev/null +++ b/event_rest_api/__init__.py @@ -0,0 +1,2 @@ +from . import pydantic_models +from . import services diff --git a/event_rest_api/__manifest__.py b/event_rest_api/__manifest__.py new file mode 100644 index 000000000..783ad6b9e --- /dev/null +++ b/event_rest_api/__manifest__.py @@ -0,0 +1,18 @@ +# Copyright 2021 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +{ + "name": "Event Rest Api", + "summary": """Add a REST API to manage events""", + "version": "14.0.1.0.0", + "license": "AGPL-3", + "author": "ACSONE SA/NV,Odoo Community Association (OCA)", + "website": "https://github.com/OCA/event", + "depends": ["base_rest", "base_rest_pydantic", "event"], + "data": [], + "external_dependencies": { + "python": [ + "pydantic", + ] + }, +} diff --git a/event_rest_api/pydantic_models/__init__.py b/event_rest_api/pydantic_models/__init__.py new file mode 100644 index 000000000..be2696013 --- /dev/null +++ b/event_rest_api/pydantic_models/__init__.py @@ -0,0 +1,9 @@ +from . import event_info +from . import event_registration_info +from . import event_registration_request +from . import event_search_filter +from . import event_stage_info +from . import event_stage_search_filter +from . import event_ticket_info +from . import event_type_info +from . import event_type_search_filter diff --git a/event_rest_api/pydantic_models/event_info.py b/event_rest_api/pydantic_models/event_info.py new file mode 100644 index 000000000..3e9581cb5 --- /dev/null +++ b/event_rest_api/pydantic_models/event_info.py @@ -0,0 +1,27 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +from datetime import datetime +from typing import List + +import pydantic + +from odoo.addons.pydantic import models, utils + +from .event_stage_info import EventStageInfo +from .event_ticket_info import EventTicketInfo +from .event_type_info import EventTypeInfo + + +class EventInfo(models.BaseModel): + id: int + name: str + date_begin: datetime + date_end: datetime + event_tickets: List[EventTicketInfo] = pydantic.Field(..., alias="event_ticket_ids") + event_type: EventTypeInfo = pydantic.Field(..., alias="event_type_id") + stage: EventStageInfo = pydantic.Field(..., alias="stage_id") + + class Config: + orm_mode = True + getter_dict = utils.GenericOdooGetter diff --git a/event_rest_api/pydantic_models/event_registration_info.py b/event_rest_api/pydantic_models/event_registration_info.py new file mode 100644 index 000000000..c60d67cbe --- /dev/null +++ b/event_rest_api/pydantic_models/event_registration_info.py @@ -0,0 +1,23 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +import pydantic + +from odoo.addons.pydantic import models, utils + +from .event_info import EventInfo +from .event_ticket_info import EventTicketInfo + + +class EventRegistrationInfo(models.BaseModel): + id: int + partner_id: int = None + firstname: str = None + lastname: str = None + email: str = None + event: EventInfo = pydantic.Field(..., alias="event_id") + event_ticket: EventTicketInfo = pydantic.Field(..., alias="event_ticket_id") + + class Config: + orm_mode = True + getter_dict = utils.GenericOdooGetter diff --git a/event_rest_api/pydantic_models/event_registration_request.py b/event_rest_api/pydantic_models/event_registration_request.py new file mode 100644 index 000000000..8b54b109a --- /dev/null +++ b/event_rest_api/pydantic_models/event_registration_request.py @@ -0,0 +1,20 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +from typing import List + +from odoo.addons.pydantic import models + + +class EventRegistrationRequest(models.BaseModel): + + firstname: str + lastname: str + email: str + phone: str = None + event_ticket_id: int = None + + +class EventRegistrationRequestList(models.BaseModel): + + event_registration_requests: List[EventRegistrationRequest] = [] diff --git a/event_rest_api/pydantic_models/event_search_filter.py b/event_rest_api/pydantic_models/event_search_filter.py new file mode 100644 index 000000000..b274219b1 --- /dev/null +++ b/event_rest_api/pydantic_models/event_search_filter.py @@ -0,0 +1,17 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +from datetime import datetime +from typing import List + +from odoo.addons.pydantic import models + + +class EventSearchFilter(models.BaseModel): + + id: int = None + name: str = None + start_after: datetime = None + end_before: datetime = None + event_type_ids: List[int] = None + stage_ids: List[int] = None diff --git a/event_rest_api/pydantic_models/event_stage_info.py b/event_rest_api/pydantic_models/event_stage_info.py new file mode 100644 index 000000000..d658d7f5a --- /dev/null +++ b/event_rest_api/pydantic_models/event_stage_info.py @@ -0,0 +1,15 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +from odoo.addons.pydantic import models, utils + + +class EventStageInfo(models.BaseModel): + id: int + name: str + sequence: int = None + pipe_end: bool = None + + class Config: + orm_mode = True + getter_dict = utils.GenericOdooGetter diff --git a/event_rest_api/pydantic_models/event_stage_search_filter.py b/event_rest_api/pydantic_models/event_stage_search_filter.py new file mode 100644 index 000000000..00ed3c176 --- /dev/null +++ b/event_rest_api/pydantic_models/event_stage_search_filter.py @@ -0,0 +1,11 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +from odoo.addons.pydantic import models + + +class EventStageSearchFilter(models.BaseModel): + + id: int = None + name: str = None + pipe_end: bool = None diff --git a/event_rest_api/pydantic_models/event_ticket_info.py b/event_rest_api/pydantic_models/event_ticket_info.py new file mode 100644 index 000000000..b6cd178cb --- /dev/null +++ b/event_rest_api/pydantic_models/event_ticket_info.py @@ -0,0 +1,20 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +from datetime import date + +from odoo.addons.pydantic import models, utils + + +class EventTicketInfo(models.BaseModel): + id: int + event_id: int + name: str + description: str = None + start_sale_date: date = None + end_sale_date: date = None + seats_available: int = None + + class Config: + orm_mode = True + getter_dict = utils.GenericOdooGetter diff --git a/event_rest_api/pydantic_models/event_type_info.py b/event_rest_api/pydantic_models/event_type_info.py new file mode 100644 index 000000000..bec74e601 --- /dev/null +++ b/event_rest_api/pydantic_models/event_type_info.py @@ -0,0 +1,13 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +from odoo.addons.pydantic import models, utils + + +class EventTypeInfo(models.BaseModel): + id: int + name: str + + class Config: + orm_mode = True + getter_dict = utils.GenericOdooGetter diff --git a/event_rest_api/pydantic_models/event_type_search_filter.py b/event_rest_api/pydantic_models/event_type_search_filter.py new file mode 100644 index 000000000..74d9520ff --- /dev/null +++ b/event_rest_api/pydantic_models/event_type_search_filter.py @@ -0,0 +1,10 @@ +# Copyright 2021 ACSONE SA/NV +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). + +from odoo.addons.pydantic import models + + +class EventTypeSearchFilter(models.BaseModel): + + id: int = None + name: str = None diff --git a/event_rest_api/readme/CONTRIBUTORS.rst b/event_rest_api/readme/CONTRIBUTORS.rst new file mode 100644 index 000000000..5914f5529 --- /dev/null +++ b/event_rest_api/readme/CONTRIBUTORS.rst @@ -0,0 +1 @@ +* Quentin Groulard diff --git a/event_rest_api/readme/DESCRIPTION.rst b/event_rest_api/readme/DESCRIPTION.rst new file mode 100644 index 000000000..1333ed77b --- /dev/null +++ b/event_rest_api/readme/DESCRIPTION.rst @@ -0,0 +1 @@ +TODO diff --git a/event_rest_api/readme/USAGE.rst b/event_rest_api/readme/USAGE.rst new file mode 100644 index 000000000..1333ed77b --- /dev/null +++ b/event_rest_api/readme/USAGE.rst @@ -0,0 +1 @@ +TODO diff --git a/event_rest_api/services/__init__.py b/event_rest_api/services/__init__.py new file mode 100644 index 000000000..73b205570 --- /dev/null +++ b/event_rest_api/services/__init__.py @@ -0,0 +1,4 @@ +from . import service +from . import event +from . import event_stage +from . import event_type diff --git a/event_rest_api/services/event.py b/event_rest_api/services/event.py new file mode 100644 index 000000000..73e6dbb7d --- /dev/null +++ b/event_rest_api/services/event.py @@ -0,0 +1,109 @@ +# Copyright 2021 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from typing import List + +from odoo import _ +from odoo.exceptions import ValidationError + +from odoo.addons.base_rest import restapi +from odoo.addons.base_rest_pydantic.restapi import PydanticModel, PydanticModelList +from odoo.addons.component.core import Component + +from ..pydantic_models.event_info import EventInfo +from ..pydantic_models.event_registration_info import EventRegistrationInfo +from ..pydantic_models.event_registration_request import ( + EventRegistrationRequest, + EventRegistrationRequestList, +) +from ..pydantic_models.event_search_filter import EventSearchFilter + + +class EventService(Component): + _inherit = "base.event.rest.service" + _name = "event.rest.service" + _usage = "event" + _expose_model = "event.event" + _description = __doc__ + + @restapi.method( + routes=[(["/"], "GET")], + output_param=PydanticModel(EventInfo), + auth="public", + ) + def get(self, _id: int) -> EventInfo: + event = self._get(_id) + return EventInfo.from_orm(event) + + def _get_search_domain(self, filters): + domain = [] + if filters.name: + domain.append(("name", "like", filters.name)) + if filters.id: + domain.append(("id", "=", filters.id)) + if filters.start_after: + domain.append(("date_begin", ">", filters.start_after)) + if filters.end_before: + domain.append(("date_end", "<", filters.end_before)) + if filters.event_type_ids: + domain.append(("event_type_id", "in", filters.event_type_ids)) + if filters.stage_ids: + domain.append(("stage_id", "in", filters.stage_ids)) + return domain + + @restapi.method( + routes=[(["/", "/search"], "GET")], + input_param=PydanticModel(EventSearchFilter), + output_param=PydanticModelList(EventInfo), + auth="public", + ) + def search(self, event_search_filter: EventSearchFilter) -> List[EventInfo]: + domain = self._get_search_domain(event_search_filter) + res: List[EventInfo] = [] + for e in self.env["event.event"].sudo().search(domain): + res.append(EventInfo.from_orm(e)) + return res + + def _prepare_event_registration_values( + self, event, event_registration_request: EventRegistrationRequest + ) -> dict: + return { + "event_id": event.id, + "partner_id": self.env.context.get("authenticated_partner_id", False), + "firstname": event_registration_request.firstname, + "lastname": event_registration_request.lastname, + "email": event_registration_request.email, + "phone": event_registration_request.phone, + "event_ticket_id": event_registration_request.event_ticket_id, + } + + @restapi.method( + routes=[(["//registration"], "POST")], + input_param=PydanticModel(EventRegistrationRequestList), + output_param=PydanticModelList(EventRegistrationInfo), + auth="public_or_default", + ) + def registration( + self, _id: int, event_registration_request_list: EventRegistrationRequestList + ) -> List[EventRegistrationInfo]: + event = self._get(_id) + if event.seats_limited: + ordered_seats = len( + event_registration_request_list.event_registration_requests + ) + if event.seats_available < ordered_seats: + raise ValidationError( + _("Not enough seats available: %s") % (event.seats_available) + ) + res: List[EventRegistrationInfo] = [] + for ( + event_registration_request + ) in event_registration_request_list.event_registration_requests: + event_registration_values = self._prepare_event_registration_values( + event, event_registration_request + ) + event_registration = self.env["event.registration"].create( + event_registration_values + ) + res.append(EventRegistrationInfo.from_orm(event_registration)) + return res diff --git a/event_rest_api/services/event_stage.py b/event_rest_api/services/event_stage.py new file mode 100644 index 000000000..b272b3e5e --- /dev/null +++ b/event_rest_api/services/event_stage.py @@ -0,0 +1,53 @@ +# Copyright 2021 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from typing import List + +from odoo.addons.base_rest import restapi +from odoo.addons.base_rest_pydantic.restapi import PydanticModel, PydanticModelList +from odoo.addons.component.core import Component + +from ..pydantic_models.event_stage_info import EventStageInfo +from ..pydantic_models.event_stage_search_filter import EventStageSearchFilter + + +class EventStageService(Component): + _inherit = "base.event.rest.service" + _name = "event.stage.rest.service" + _usage = "event_stage" + _expose_model = "event.stage" + _description = __doc__ + + @restapi.method( + routes=[(["/"], "GET")], + output_param=PydanticModel(EventStageInfo), + auth="public", + ) + def get(self, _id: int) -> EventStageInfo: + event_stage = self._get(_id) + return EventStageInfo.from_orm(event_stage) + + def _get_search_domain(self, filters): + domain = [] + if filters.name: + domain.append(("name", "like", filters.name)) + if filters.id: + domain.append(("id", "=", filters.id)) + if filters.pipe_end: + domain.append(("pipe_end", "=", filters.pipe_end)) + return domain + + @restapi.method( + routes=[(["/", "/search"], "GET")], + input_param=PydanticModel(EventStageSearchFilter), + output_param=PydanticModelList(EventStageInfo), + auth="public", + ) + def search( + self, event_stage_search_filter: EventStageSearchFilter + ) -> List[EventStageInfo]: + domain = self._get_search_domain(event_stage_search_filter) + res: List[EventStageInfo] = [] + for e in self.env["event.stage"].sudo().search(domain): + res.append(EventStageInfo.from_orm(e)) + return res diff --git a/event_rest_api/services/event_type.py b/event_rest_api/services/event_type.py new file mode 100644 index 000000000..3172b138f --- /dev/null +++ b/event_rest_api/services/event_type.py @@ -0,0 +1,51 @@ +# Copyright 2021 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from typing import List + +from odoo.addons.base_rest import restapi +from odoo.addons.base_rest_pydantic.restapi import PydanticModel, PydanticModelList +from odoo.addons.component.core import Component + +from ..pydantic_models.event_type_info import EventTypeInfo +from ..pydantic_models.event_type_search_filter import EventTypeSearchFilter + + +class EventTypeService(Component): + _inherit = "base.event.rest.service" + _name = "event.type.rest.service" + _usage = "event_type" + _expose_model = "event.type" + _description = __doc__ + + @restapi.method( + routes=[(["/"], "GET")], + output_param=PydanticModel(EventTypeInfo), + auth="public", + ) + def get(self, _id: int) -> EventTypeInfo: + event_type = self._get(_id) + return EventTypeInfo.from_orm(event_type) + + def _get_search_domain(self, filters): + domain = [] + if filters.name: + domain.append(("name", "like", filters.name)) + if filters.id: + domain.append(("id", "=", filters.id)) + return domain + + @restapi.method( + routes=[(["/", "/search"], "GET")], + input_param=PydanticModel(EventTypeSearchFilter), + output_param=PydanticModelList(EventTypeInfo), + auth="public", + ) + def search( + self, event_type_search_filter: EventTypeSearchFilter + ) -> List[EventTypeInfo]: + domain = self._get_search_domain(event_type_search_filter) + res: List[EventTypeInfo] = [] + for e in self.env["event.type"].sudo().search(domain): + res.append(EventTypeInfo.from_orm(e)) + return res diff --git a/event_rest_api/services/service.py b/event_rest_api/services/service.py new file mode 100644 index 000000000..0ee9690bd --- /dev/null +++ b/event_rest_api/services/service.py @@ -0,0 +1,24 @@ +# Copyright 2021 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import _ +from odoo.exceptions import MissingError + +from odoo.addons.component.core import AbstractComponent + + +class BaseEventService(AbstractComponent): + _inherit = "base.rest.service" + _name = "base.event.rest.service" + _collection = "event.rest.services" + _expose_model = None + + def _get(self, _id): + domain = [("id", "=", _id)] + record = self.env[self._expose_model].search(domain) + if not record: + raise MissingError( + _("The record %s %s does not exist") % (self._expose_model, _id) + ) + else: + return record diff --git a/event_rest_api/static/description/icon.png b/event_rest_api/static/description/icon.png new file mode 100644 index 000000000..3a0328b51 Binary files /dev/null and b/event_rest_api/static/description/icon.png differ diff --git a/event_rest_api/static/description/index.html b/event_rest_api/static/description/index.html new file mode 100644 index 000000000..9f30ef0e5 --- /dev/null +++ b/event_rest_api/static/description/index.html @@ -0,0 +1,424 @@ + + + + + + +Event Rest Api + + + +
+

Event Rest Api

+ + +

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

+

TODO

+

Table of contents

+ +
+

Usage

+

TODO

+
+
+

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/event project on GitHub.

+

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

+
+
+
+ + diff --git a/event_rest_api/tests/__init__.py b/event_rest_api/tests/__init__.py new file mode 100644 index 000000000..8d73378b6 --- /dev/null +++ b/event_rest_api/tests/__init__.py @@ -0,0 +1 @@ +from . import test_event diff --git a/event_rest_api/tests/test_event.py b/event_rest_api/tests/test_event.py new file mode 100644 index 000000000..6121781b2 --- /dev/null +++ b/event_rest_api/tests/test_event.py @@ -0,0 +1,43 @@ +# Copyright 2021 ACSONE SA/NV +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from datetime import datetime + +from odoo.http import request + +from odoo.addons.base_rest.controllers.main import _PseudoCollection +from odoo.addons.base_rest.tests.common import BaseRestCase +from odoo.addons.component.core import WorkContext +from odoo.addons.pydantic.tests.common import PydanticMixin + + +class EventCase(BaseRestCase, PydanticMixin): + @classmethod + def setUpClass(cls): + super().setUpClass() + collection = _PseudoCollection("event.rest.services", cls.env) + cls.services_env = WorkContext( + model_name="rest.service.registration", + collection=collection, + request=request, + ) + cls.service = cls.services_env.component(usage="event") + cls.event = cls.env["event.event"].create( + { + "name": "Test Event", + "date_begin": datetime.now(), + "date_end": datetime.now(), + } + ) + cls.setUpPydantic() + + # pylint: disable=W8106 + def setUp(self): + # resolve an inheritance issue (common.TransactionCase does not call + # super) + BaseRestCase.setUp(self) + PydanticMixin.setUp(self) + + def test_get_event(self): + res = self.service.dispatch("get", self.event.id) + self.assertEqual(res["name"], "Test Event") diff --git a/oca_dependencies.txt b/oca_dependencies.txt index ca3c726ba..b2e375a15 100644 --- a/oca_dependencies.txt +++ b/oca_dependencies.txt @@ -1 +1,2 @@ # See https://github.com/OCA/odoo-community.org/blob/master/website/Contribution/CONTRIBUTING.rst#oca_dependencies-txt +rest-framework https://github.com/acsone/rest-framework 14.0-base-rest-pydantic-lmi diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 000000000..e582a798e --- /dev/null +++ b/requirements.txt @@ -0,0 +1,2 @@ +# generated from manifests external_dependencies +pydantic diff --git a/setup/event_rest_api/odoo/addons/event_rest_api b/setup/event_rest_api/odoo/addons/event_rest_api new file mode 120000 index 000000000..2bb9481b4 --- /dev/null +++ b/setup/event_rest_api/odoo/addons/event_rest_api @@ -0,0 +1 @@ +../../../../event_rest_api \ No newline at end of file diff --git a/setup/event_rest_api/setup.py b/setup/event_rest_api/setup.py new file mode 100644 index 000000000..28c57bb64 --- /dev/null +++ b/setup/event_rest_api/setup.py @@ -0,0 +1,6 @@ +import setuptools + +setuptools.setup( + setup_requires=['setuptools-odoo'], + odoo_addon=True, +)