diff --git a/connector_extension_wordpress/README.rst b/connector_extension_wordpress/README.rst new file mode 100644 index 000000000..16ccce2ee --- /dev/null +++ b/connector_extension_wordpress/README.rst @@ -0,0 +1,65 @@ +============================= +Connector Extension Wordpress +============================= + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:5acc311916374a5c8b68547091798eb4307996ed461862d35d5563b1c1b6bdd1 + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |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-LGPL--3-blue.png + :target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html + :alt: License: LGPL-3 +.. |badge3| image:: https://img.shields.io/badge/github-NuoBiT%2Fodoo--addons-lightgray.png?logo=github + :target: https://github.com/NuoBiT/odoo-addons/tree/18.0/connector_extension_wordpress + :alt: NuoBiT/odoo-addons + +|badge1| |badge2| |badge3| + +This module extends the connector extension module to add support for +Wordpress + +**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 to smash it by providing a detailed and welcomed +`feedback `_. + +Do not contact contributors directly about support or help with technical issues. + +Credits +======= + +Authors +------- + +* NuoBiT Solutions SL + +Contributors +------------ + +- `NuoBiT `__: + + - Kilian Niubo kniubo@nuobit.com + - Eric Antones eantones@nuobit.com + - Deniz Gallo dgallo@nuobit.com + +Maintainers +----------- + +This module is part of the `NuoBiT/odoo-addons `_ project on GitHub. + +You are welcome to contribute. diff --git a/connector_extension_wordpress/__init__.py b/connector_extension_wordpress/__init__.py new file mode 100644 index 000000000..1377f57f5 --- /dev/null +++ b/connector_extension_wordpress/__init__.py @@ -0,0 +1 @@ +from . import components diff --git a/connector_extension_wordpress/__manifest__.py b/connector_extension_wordpress/__manifest__.py new file mode 100644 index 000000000..6c14616af --- /dev/null +++ b/connector_extension_wordpress/__manifest__.py @@ -0,0 +1,16 @@ +# Copyright NuoBiT Solutions - Kilian Niubo +# Copyright NuoBiT Solutions - Eric Antones +# Copyright 2026 NuoBiT Solutions SL - Deniz Gallo +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html) + +{ + "name": "Connector Extension Wordpress", + "summary": "This module extends the connector extension module " + "to add support for Wordpress", + "version": "18.0.1.0.0", + "author": "NuoBiT Solutions SL", + "license": "LGPL-3", + "category": "Connector", + "website": "https://github.com/NuoBiT/odoo-addons", + "depends": ["connector_extension"], +} diff --git a/connector_extension_wordpress/components/__init__.py b/connector_extension_wordpress/components/__init__.py new file mode 100644 index 000000000..f502287fe --- /dev/null +++ b/connector_extension_wordpress/components/__init__.py @@ -0,0 +1 @@ +from . import adapter diff --git a/connector_extension_wordpress/components/adapter.py b/connector_extension_wordpress/components/adapter.py new file mode 100644 index 000000000..add68ec98 --- /dev/null +++ b/connector_extension_wordpress/components/adapter.py @@ -0,0 +1,230 @@ +# Copyright NuoBiT Solutions - Kilian Niubo +# Copyright 2026 NuoBiT Solutions SL - Deniz Gallo +# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html) + +import json +import logging + +import requests + +from odoo import _ +from odoo.exceptions import ValidationError + +from odoo.addons.component.core import AbstractComponent +from odoo.addons.connector.exception import RetryableJobError + +_logger = logging.getLogger(__name__) + + +class ConnectorExtensionWordpressAdapterCRUD(AbstractComponent): + _name = "connector.extension.wordpress.adapter.crud" + _inherit = "connector.extension.adapter.crud" + + def _exec(self, op, resource, *args, **kwargs): + func = getattr(self, f"_exec_{op}") + return func(resource, *args, **kwargs) + + def _manage_error_codes( + self, op, res_data, res, resource, *args, raise_on_error=True, **kwargs + ): + if not res.ok: + error_message = None + if res.status_code == 404: + if res_data.get("code") == "rest_post_invalid_id": + error_message = _( + "Error: '%(message)s'. Probably the %(resource)s has been " + "removed from WordPress. " + "If it's the case, try to remove the binding of the %(model)s." + ) % { + "message": res_data.get("message"), + "resource": resource, + "model": self.model._name, + } + elif res.status_code == 500: + if res_data.get("code") == "rest_upload_sideload_error": + error_message = _( + "Error: '%(message)s'. Probably the image or document " + "is uploaded with bad format. " + "Please, review on database: %(resource)s" + ) % { + "message": res_data["message"], + "resource": kwargs["headers"], + } + if not error_message: + error_message = _("Error: %(message)s, Resource: %(resource)s") % { + "message": res_data, + "resource": resource, + } + if raise_on_error: + raise ValidationError(error_message) + return error_message + return res_data + + def _exec_wp_call(self, op, resource, *args, **kwargs): + url = self.backend_record.url + "/wp-json/wp/v2/" + resource + func = getattr(requests, op) + try: + res = func(url, *args, **kwargs) + res_data = res.json() + res_data = self._manage_error_codes( + op, res_data, res, resource, *args, **kwargs + ) + result = { + "ok": res.ok, + "status_code": res.status_code, + "headers": res.headers, + "data": res_data, + } + except requests.exceptions.ConnectionError as e: + raise RetryableJobError(_("Error connecting to WordPress: %s") % e) from e + except json.JSONDecodeError as e: + raise ValidationError( + _("Error decoding json WordPress response: %(message)s\n%(response)s") + % { + "message": e, + "response": res.text if res else _("No response"), + } + ) from e + return result + + def _get_search_fields(self): + return ["modified_after", "offset", "per_page", "page"] + + def get_total_items(self, resource, domain=None): + filters_values = self._get_search_fields() + real_domain, common_domain = self._extract_domain_clauses( + domain, filters_values + ) + params = self._domain_to_normalized_dict(real_domain) + # TODO: make an optimization to get the total items and use the result + params["per_page"] = 1 + result = self._exec_wp_call( + "get", + resource, + auth=( + self.backend_record.username, + self.backend_record.application_password, + ), + params=params, + verify=self.backend_record.verify_ssl, + ) + + total_items_header = result["headers"]._store.get("x-wp-total") + if total_items_header: + total_items_header = int(total_items_header[1]) + else: + # WordPress returns a dict if the response is a single item + if not isinstance(result["data"], list): + result["data"] = [result["data"]] + total_items_header = len(result["data"]) + return total_items_header + + # TODO: REVIEW: Unify with connector_extension_woocommerce + def _exec_get(self, resource, *args, **kwargs): + if resource == "system_status": + return self._exec_wp_call( + "get", + resource, + *args, + auth=( + self.backend_record.username, + self.backend_record.application_password, + ), + verify=self.backend_record.verify_ssl, + **kwargs, + ) + # WooCommerce has the parameter next on the response headers + # to get the next page but we can't use it because if we use + # the offset, the next page will have the same items as the first page. + # It looks like a bug in WooCommerce API. + domain = [] + if "domain" in kwargs: + domain = kwargs.pop("domain") + search_fields = self._get_search_fields() + real_domain, common_domain = self._extract_domain_clauses(domain, search_fields) + params = self._domain_to_normalized_dict(real_domain) + if "limit" in kwargs: + limit = kwargs.pop("limit") + else: + limit = self.get_total_items(resource, domain) + params["offset"] = ( + kwargs.pop("offset") if "offset" in kwargs and "offset" not in params else 0 + ) + page_size = self.backend_record.page_size + params["per_page"] = page_size if page_size > 0 else 100 + data = [] + while len(data) < limit: + if page_size > limit - len(data): + params["per_page"] = limit - len(data) + res = self._exec_wp_call( + "get", + resource, + *args, + params=params, + auth=( + self.backend_record.username, + self.backend_record.application_password, + ), + verify=self.backend_record.verify_ssl, + **kwargs, + ) + # WooCommerce returns a dict if the response is a single item + if not isinstance(res["data"], list): + res["data"] = [res["data"]] + data += res["data"] + params["offset"] += len(res["data"]) + return self._filter(data, common_domain) + + def _exec_post(self, resource, *args, **kwargs): + auth = (self.backend_record.username, self.backend_record.application_password) + if "wordpress_backend_id" in self.backend_record: + backend = self.backend_record.wordpress_backend_id + auth = (backend.username, backend.application_password) + data_aux = kwargs.pop("data", {}) + headers = data_aux.pop("headers", {}) + data = data_aux.pop("data", {}) + res = self._exec_wp_call( + "post", + resource, + data=data, + params=data_aux, + headers=headers, + auth=auth, + verify=self.backend_record.verify_ssl, + ) + + return res["data"] + + def _exec_put(self, resource, *args, **kwargs): + auth = (self.backend_record.username, self.backend_record.application_password) + if "wordpress_backend_id" in self.backend_record: + backend = self.backend_record.wordpress_backend_id + auth = (backend.username, backend.application_password) + data_aux = kwargs.pop("data", {}) + headers = data_aux.pop("headers", {}) + data = data_aux.pop("data", {}) + res = self._exec_wp_call( + "put", + resource, + *args, + data=data, + params=data_aux, + headers=headers, + auth=auth, + verify=self.backend_record.verify_ssl, + **kwargs, + ) + return res["data"] + + def _exec_delete(self, resource, *args, **kwargs): + raise NotImplementedError() + + def _exec_options(self, resource, *args, **kwargs): + raise NotImplementedError() + + def get_version(self): + settings = self._exec("get", "settings") + if settings and settings[0].get("title"): + return "Wordpress '{}' connected".format(settings[0].get("title")) + else: + raise ValidationError(_("Wordpress not connected")) diff --git a/connector_extension_wordpress/i18n/ca.po b/connector_extension_wordpress/i18n/ca.po new file mode 100644 index 000000000..c7e5ad4ca --- /dev/null +++ b/connector_extension_wordpress/i18n/ca.po @@ -0,0 +1,64 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * connector_extension_wordpress +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 14.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2024-10-22 12:56+0000\n" +"PO-Revision-Date: 2024-10-22 12:56+0000\n" +"Last-Translator: \n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "Error connecting to WordPress: %s" +msgstr "Error al conectar amb WordPress: %s" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "" +"Error decoding json WordPress response: %s\n" +"%s" +msgstr "" +"Error al descodificar la resposta json de WordPress: %s\n" +"%s" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "Error: %s, Resource: %s" +msgstr "Error: %s, Recurs: %s" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "" +"Error: '%s'. Probably the %s has been removed from WordPress. If it's the " +"case, try to remove the binding of the %s." +msgstr "" +"Error: '%s'. Probablement el %s ha estat eliminat de WordPress. Si és el " +"cas, intenta eliminar la vinculació del %s." + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "" +"Error: '%s'. Probably the image or document is uploaded with bad format. " +"Please, review on database: %s" +msgstr "" +"Error: '%s'. Probablement la imatge o el document s'ha pujat amb un format " +"incorrecte. Si us plau, revisa a la base de dades: %s" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "Wordpress not connected" +msgstr "WordPress no connectat" diff --git a/connector_extension_wordpress/i18n/es.po b/connector_extension_wordpress/i18n/es.po new file mode 100644 index 000000000..3e0c209a5 --- /dev/null +++ b/connector_extension_wordpress/i18n/es.po @@ -0,0 +1,64 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * connector_extension_wordpress +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 14.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2024-10-22 12:54+0000\n" +"PO-Revision-Date: 2024-10-22 12:54+0000\n" +"Last-Translator: \n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "Error connecting to WordPress: %s" +msgstr "Error al conectar con WordPress: %s" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "" +"Error decoding json WordPress response: %s\n" +"%s" +msgstr "" +"Error al decodificar la respuesta json de WordPress: %s\n" +"%s" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "Error: %s, Resource: %s" +msgstr "Error: %s, Recurso: %s" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "" +"Error: '%s'. Probably the %s has been removed from WordPress. If it's the " +"case, try to remove the binding of the %s." +msgstr "" +"Error: '%s'. Probablemente el %s ha sido eliminado de WordPress. Si es el " +"caso, intente eliminar la vinculación del %s." + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "" +"Error: '%s'. Probably the image or document is uploaded with bad format. " +"Please, review on database: %s" +msgstr "" +"Error: '%s'. Probablemente la imagen o el documento se ha subido con un " +"formato incorrecto. Por favor, revise en la base de datos: %s" + +#. module: connector_extension_wordpress +#: code:addons/connector_extension_wordpress/components/adapter.py:0 +#, python-format +msgid "Wordpress not connected" +msgstr "WordPress no conectado" diff --git a/connector_extension_wordpress/pyproject.toml b/connector_extension_wordpress/pyproject.toml new file mode 100644 index 000000000..4231d0ccc --- /dev/null +++ b/connector_extension_wordpress/pyproject.toml @@ -0,0 +1,3 @@ +[build-system] +requires = ["whool"] +build-backend = "whool.buildapi" diff --git a/connector_extension_wordpress/readme/CONTRIBUTORS.md b/connector_extension_wordpress/readme/CONTRIBUTORS.md new file mode 100644 index 000000000..0355cecbc --- /dev/null +++ b/connector_extension_wordpress/readme/CONTRIBUTORS.md @@ -0,0 +1,4 @@ +- [NuoBiT](https://www.nuobit.com): + - Kilian Niubo + - Eric Antones + - Deniz Gallo diff --git a/connector_extension_wordpress/readme/DESCRIPTION.md b/connector_extension_wordpress/readme/DESCRIPTION.md new file mode 100644 index 000000000..6340d617e --- /dev/null +++ b/connector_extension_wordpress/readme/DESCRIPTION.md @@ -0,0 +1,2 @@ +This module extends the connector extension module to add support for +Wordpress diff --git a/connector_extension_wordpress/static/description/icon.png b/connector_extension_wordpress/static/description/icon.png new file mode 100644 index 000000000..1cd641e79 Binary files /dev/null and b/connector_extension_wordpress/static/description/icon.png differ diff --git a/connector_extension_wordpress/static/description/index.html b/connector_extension_wordpress/static/description/index.html new file mode 100644 index 000000000..5b4765460 --- /dev/null +++ b/connector_extension_wordpress/static/description/index.html @@ -0,0 +1,422 @@ + + + + + +Connector Extension Wordpress + + + +
+

Connector Extension Wordpress

+ + +

Beta License: LGPL-3 NuoBiT/odoo-addons

+

This module extends the connector extension module to add support for +Wordpress

+

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 to smash it by providing a detailed and welcomed +feedback.

+

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

+
+
+

Credits

+
+

Authors

+
    +
  • NuoBiT Solutions SL
  • +
+
+
+

Contributors

+ +
+
+

Maintainers

+

This module is part of the NuoBiT/odoo-addons project on GitHub.

+

You are welcome to contribute.

+
+
+
+ + diff --git a/setup/connector_extension_wordpress/odoo/addons/connector_extension_wordpress b/setup/connector_extension_wordpress/odoo/addons/connector_extension_wordpress new file mode 120000 index 000000000..aa5befdcc --- /dev/null +++ b/setup/connector_extension_wordpress/odoo/addons/connector_extension_wordpress @@ -0,0 +1 @@ +../../../../connector_extension_wordpress \ No newline at end of file diff --git a/setup/connector_extension_wordpress/setup.py b/setup/connector_extension_wordpress/setup.py new file mode 100644 index 000000000..28c57bb64 --- /dev/null +++ b/setup/connector_extension_wordpress/setup.py @@ -0,0 +1,6 @@ +import setuptools + +setuptools.setup( + setup_requires=['setuptools-odoo'], + odoo_addon=True, +)