[ADD] dependant modules
This commit is contained in:
parent
fa3d921e2d
commit
e8bc408a7a
|
|
@ -0,0 +1,106 @@
|
||||||
|
======================
|
||||||
|
Verify email at signup
|
||||||
|
======================
|
||||||
|
|
||||||
|
.. !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
|
!! 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%2Fserver--auth-lightgray.png?logo=github
|
||||||
|
:target: https://github.com/OCA/server-auth/tree/13.0/auth_signup_verify_email
|
||||||
|
:alt: OCA/server-auth
|
||||||
|
.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png
|
||||||
|
:target: https://translation.odoo-community.org/projects/server-auth-13-0/server-auth-13-0-auth_signup_verify_email
|
||||||
|
:alt: Translate me on Weblate
|
||||||
|
.. |badge5| image:: https://img.shields.io/badge/runbot-Try%20me-875A7B.png
|
||||||
|
:target: https://runbot.odoo-community.org/runbot/251/13.0
|
||||||
|
:alt: Try me on Runbot
|
||||||
|
|
||||||
|
|badge1| |badge2| |badge3| |badge4| |badge5|
|
||||||
|
|
||||||
|
This module extends the functionality of public sign up to force users to
|
||||||
|
provide a valid email address.
|
||||||
|
|
||||||
|
To achieve this, users are not required to provide a password at
|
||||||
|
sign up: they are asked for only at first login attempt.
|
||||||
|
|
||||||
|
**Table of contents**
|
||||||
|
|
||||||
|
.. contents::
|
||||||
|
:local:
|
||||||
|
|
||||||
|
Installation
|
||||||
|
============
|
||||||
|
|
||||||
|
* Install `email_validator <https://pypi.org/project/email-validator/>`_
|
||||||
|
with ``pip install email_validator`` or equivalent.
|
||||||
|
|
||||||
|
Configuration
|
||||||
|
=============
|
||||||
|
|
||||||
|
To configure this module, you need to:
|
||||||
|
|
||||||
|
* `Properly configure your outgoing email server(s)
|
||||||
|
<https://www.odoo.com/forum/help-1/question/how-to-configure-email-gateway-282#answer_290>`_.
|
||||||
|
* Go to *Settings > General Settings -> General settings*, search for
|
||||||
|
the *Users* section and enable *Free sign up* in *Customer account*.
|
||||||
|
|
||||||
|
Usage
|
||||||
|
=====
|
||||||
|
|
||||||
|
To use this module, you need to:
|
||||||
|
|
||||||
|
* Log out.
|
||||||
|
* `Sign up </web/signup>`_ with a valid email.
|
||||||
|
|
||||||
|
Bug Tracker
|
||||||
|
===========
|
||||||
|
|
||||||
|
Bugs are tracked on `GitHub Issues <https://github.com/OCA/server-auth/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 <https://github.com/OCA/server-auth/issues/new?body=module:%20auth_signup_verify_email%0Aversion:%2013.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
|
||||||
|
|
||||||
|
Do not contact contributors directly about support or help with technical issues.
|
||||||
|
|
||||||
|
Credits
|
||||||
|
=======
|
||||||
|
|
||||||
|
Authors
|
||||||
|
~~~~~~~
|
||||||
|
|
||||||
|
* Antiun Ingeniería S.L.
|
||||||
|
* Tecnativa
|
||||||
|
|
||||||
|
Contributors
|
||||||
|
~~~~~~~~~~~~
|
||||||
|
|
||||||
|
* Rafael Blasco <rafaelbn@antiun.com>
|
||||||
|
* Jairo Llopis <yajo.sk8@gmail.com>
|
||||||
|
* Simone Orsi <simone.orsi@camptocamp.com>
|
||||||
|
* Alexandre Díaz <alexandre.diaz@tecnativa.com>
|
||||||
|
* Eugene Molotov <molotov@it-projects.info>
|
||||||
|
|
||||||
|
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/server-auth <https://github.com/OCA/server-auth/tree/13.0/auth_signup_verify_email>`_ project on GitHub.
|
||||||
|
|
||||||
|
You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.
|
||||||
|
|
@ -0,0 +1,4 @@
|
||||||
|
# Copyright 2015 Antiun Ingeniería, S.L.
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||||
|
|
||||||
|
from . import controllers
|
||||||
|
|
@ -0,0 +1,17 @@
|
||||||
|
# Copyright 2015 Antiun Ingeniería, S.L.
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||||
|
{
|
||||||
|
"name": "Verify email at signup",
|
||||||
|
"summary": "Force uninvited users to use a good email for signup",
|
||||||
|
"version": "14.0.1.0.0",
|
||||||
|
"category": "Authentication",
|
||||||
|
"website": "https://github.com/OCA/server-auth",
|
||||||
|
"author": "Antiun Ingeniería S.L., "
|
||||||
|
"Tecnativa, "
|
||||||
|
"Odoo Community Association (OCA)",
|
||||||
|
"license": "AGPL-3",
|
||||||
|
"depends": ["auth_signup"],
|
||||||
|
"external_dependencies": {"python": ["lxml", "email_validator"]},
|
||||||
|
"data": ["views/signup.xml"],
|
||||||
|
"installable": True,
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,4 @@
|
||||||
|
# Copyright 2015 Antiun Ingeniería, S.L.
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||||
|
|
||||||
|
from . import main
|
||||||
|
|
@ -0,0 +1,78 @@
|
||||||
|
# Copyright 2015 Antiun Ingeniería, S.L.
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from email_validator import EmailSyntaxError, EmailUndeliverableError, validate_email
|
||||||
|
|
||||||
|
from odoo import _
|
||||||
|
from odoo.http import request, route
|
||||||
|
|
||||||
|
from odoo.addons.auth_signup.controllers.main import AuthSignupHome
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class SignupVerifyEmail(AuthSignupHome):
|
||||||
|
@route()
|
||||||
|
def web_auth_signup(self, *args, **kw):
|
||||||
|
if request.params.get("login") and not request.params.get("password"):
|
||||||
|
return self.passwordless_signup()
|
||||||
|
return super().web_auth_signup(*args, **kw)
|
||||||
|
|
||||||
|
def passwordless_signup(self):
|
||||||
|
values = request.params
|
||||||
|
qcontext = self.get_auth_signup_qcontext()
|
||||||
|
|
||||||
|
# Check good format of e-mail
|
||||||
|
try:
|
||||||
|
validate_email(values.get("login", ""))
|
||||||
|
except EmailSyntaxError as error:
|
||||||
|
qcontext["error"] = getattr(
|
||||||
|
error, "message", _("That does not seem to be an email address."),
|
||||||
|
)
|
||||||
|
return request.render("auth_signup.signup", qcontext)
|
||||||
|
except EmailUndeliverableError as error:
|
||||||
|
qcontext["error"] = str(error)
|
||||||
|
return request.render("auth_signup.signup", qcontext)
|
||||||
|
except Exception as error:
|
||||||
|
qcontext["error"] = str(error)
|
||||||
|
return request.render("auth_signup.signup", qcontext)
|
||||||
|
if not values.get("email"):
|
||||||
|
values["email"] = values.get("login")
|
||||||
|
|
||||||
|
# preserve user lang
|
||||||
|
values["lang"] = request.context.get("lang", "")
|
||||||
|
|
||||||
|
# remove values that could raise "Invalid field '*' on model 'res.users'"
|
||||||
|
values.pop("redirect", "")
|
||||||
|
values.pop("token", "")
|
||||||
|
|
||||||
|
# Remove password
|
||||||
|
values["password"] = ""
|
||||||
|
sudo_users = request.env["res.users"].with_context(create_user=True).sudo()
|
||||||
|
|
||||||
|
try:
|
||||||
|
with request.cr.savepoint():
|
||||||
|
sudo_users.signup(values, qcontext.get("token"))
|
||||||
|
sudo_users.reset_password(values.get("login"))
|
||||||
|
except Exception as error:
|
||||||
|
# Duplicate key or wrong SMTP settings, probably
|
||||||
|
_logger.exception(error)
|
||||||
|
if (
|
||||||
|
request.env["res.users"]
|
||||||
|
.sudo()
|
||||||
|
.search([("login", "=", qcontext.get("login"))])
|
||||||
|
):
|
||||||
|
qcontext["error"] = _(
|
||||||
|
"Another user is already registered using this email" " address."
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
# Agnostic message for security
|
||||||
|
qcontext["error"] = _(
|
||||||
|
"Something went wrong, please try again later or" " contact us."
|
||||||
|
)
|
||||||
|
return request.render("auth_signup.signup", qcontext)
|
||||||
|
|
||||||
|
qcontext["message"] = _("Check your email to activate your account!")
|
||||||
|
return request.render("auth_signup.reset_password", qcontext)
|
||||||
|
|
@ -0,0 +1,42 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 13.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"PO-Revision-Date: 2021-01-06 13:44+0000\n"
|
||||||
|
"Last-Translator: Rachid Al Assir <rachidalassir@gmail.com>\n"
|
||||||
|
"Language-Team: none\n"
|
||||||
|
"Language: ar\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=6; plural=n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : n%100>=3 "
|
||||||
|
"&& n%100<=10 ? 3 : n%100>=11 ? 4 : 5;\n"
|
||||||
|
"X-Generator: Weblate 4.3.2\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr "هنالك مستخدم أخر مسجل بهذا البريد الإلكتروني."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "تحقق من بريدك الإلكتروني لتفعيل حسابك!"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr "حدث خطأ ما، يرجى المحاولة مرة أخرى لاحقًا أو الاتصال بنا."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "لا يبدو أن هذا عنوان بريد إلكتروني."
|
||||||
|
|
@ -0,0 +1,38 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 13.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \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: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr ""
|
||||||
|
|
@ -0,0 +1,45 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Rudolf Schnapka <rs@techno-flex.de>, 2016
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: server-tools (9.0)\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2016-06-09 12:31+0000\n"
|
||||||
|
"PO-Revision-Date: 2016-05-31 14:47+0000\n"
|
||||||
|
"Last-Translator: OCA Transbot <transbot@odoo-community.org>\n"
|
||||||
|
"Language-Team: German (http://www.transifex.com/oca/OCA-server-tools-9-0/"
|
||||||
|
"language/de/)\n"
|
||||||
|
"Language: de\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Prüfen Sie Ihre Email, um Ihr Konto zu aktivieren."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr ""
|
||||||
|
"Etwas ist schief gelaufen, bitte später nochmal versuchen oden an uns wenden."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Das schein keine Email-Adresse zu sein."
|
||||||
|
|
@ -0,0 +1,44 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Antonio Trueba, 2016
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: server-tools (9.0)\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2016-06-09 12:31+0000\n"
|
||||||
|
"PO-Revision-Date: 2016-05-31 14:47+0000\n"
|
||||||
|
"Last-Translator: OCA Transbot <transbot@odoo-community.org>\n"
|
||||||
|
"Language-Team: Spanish (http://www.transifex.com/oca/OCA-server-tools-9-0/"
|
||||||
|
"language/es/)\n"
|
||||||
|
"Language: es\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Revise su correo para activar su cuenta."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr "Algo ha ido mal, por favor inténtelo de nuevo más tarde o contáctenos."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Eso no parece una dirección de email válida."
|
||||||
|
|
@ -0,0 +1,44 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Alexandre Papin <papin.alexandre@me.com>, 2017
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 10.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2017-03-18 02:19+0000\n"
|
||||||
|
"PO-Revision-Date: 2017-03-18 02:19+0000\n"
|
||||||
|
"Last-Translator: Alexandre Papin <papin.alexandre@me.com>, 2017\n"
|
||||||
|
"Language-Team: French (https://www.transifex.com/oca/teams/23907/fr/)\n"
|
||||||
|
"Language: fr\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=2; plural=(n > 1);\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Un email vous à été envoyé pour activer votre compte."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr ""
|
||||||
|
"Une erreur est survenue, veuillez réessayer plus tard ou contactez nous."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Veuillez fournir une adresse email valide."
|
||||||
|
|
@ -0,0 +1,45 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Bole <bole@dajmi5.com>, 2017
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 10.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2017-05-10 00:47+0000\n"
|
||||||
|
"PO-Revision-Date: 2019-11-13 17:34+0000\n"
|
||||||
|
"Last-Translator: Bole <bole@dajmi5.com>\n"
|
||||||
|
"Language-Team: Croatian (https://www.transifex.com/oca/teams/23907/hr/)\n"
|
||||||
|
"Language: hr\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n"
|
||||||
|
"%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
|
||||||
|
"X-Generator: Weblate 3.8\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr "Drugi korisnik je već registriran sa ovom mail adresom."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Provjerite svoj e-mail za aktivaciu računa!"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr "Nešto nije u redu, molimo pokušajte ponovo ili nas kontaktirajte."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Ovo ne izgleda kao e-mail adresa."
|
||||||
|
|
@ -0,0 +1,47 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Bole <bole@dajmi5.com>, 2016
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: server-tools (9.0)\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2016-06-29 00:48+0000\n"
|
||||||
|
"PO-Revision-Date: 2016-06-13 08:38+0000\n"
|
||||||
|
"Last-Translator: Bole <bole@dajmi5.com>\n"
|
||||||
|
"Language-Team: Croatian (Croatia) (http://www.transifex.com/oca/OCA-server-"
|
||||||
|
"tools-9-0/language/hr_HR/)\n"
|
||||||
|
"Language: hr_HR\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n"
|
||||||
|
"%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Provjerite svoj mail za aktiviranje računa"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr ""
|
||||||
|
"Nešto nije u redu, molimo pokušajte se prijaviti kasnije ili nas "
|
||||||
|
"kontaktirajte."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Ovo ne izgleda kao valjana mail adresa."
|
||||||
|
|
@ -0,0 +1,46 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Paolo Valier, 2016
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: server-tools (9.0)\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2016-06-09 12:31+0000\n"
|
||||||
|
"PO-Revision-Date: 2016-05-31 14:47+0000\n"
|
||||||
|
"Last-Translator: OCA Transbot <transbot@odoo-community.org>\n"
|
||||||
|
"Language-Team: Italian (http://www.transifex.com/oca/OCA-server-tools-9-0/"
|
||||||
|
"language/it/)\n"
|
||||||
|
"Language: it\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Controlla la tua email per attivare il tuo account!"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr ""
|
||||||
|
"Qualcosa non è funzionato, prego provare più tardi altrimenti mettiti in "
|
||||||
|
"contatto con noi."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Questo non sembra essere un indirizzo email valido."
|
||||||
|
|
@ -0,0 +1,47 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: server-tools (8.0)\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2015-11-27 14:01+0000\n"
|
||||||
|
"PO-Revision-Date: 2019-09-03 03:23+0000\n"
|
||||||
|
"Last-Translator: Rodrigo Macedo <rmsolucoeseminformatic4@gmail.com>\n"
|
||||||
|
"Language-Team: Portuguese (Brazil) (http://www.transifex.com/oca/OCA-server-"
|
||||||
|
"tools-8-0/language/pt_BR/)\n"
|
||||||
|
"Language: pt_BR\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=2; plural=n > 1;\n"
|
||||||
|
"X-Generator: Weblate 3.8\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr "Outro usuário já está registrado usando este endereço de email."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Verifique seu e-mail para ativar sua conta!"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr "Ocorreu um erro. Tente novamente mais tarde ou entre em contato."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Esse não parece ser um endereço de email."
|
||||||
|
|
||||||
|
#~ msgid "email"
|
||||||
|
#~ msgstr "email"
|
||||||
|
|
@ -0,0 +1,45 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Daniel Schweiger <danielcccasle@gmail.com>, 2017
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 10.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2017-06-22 01:11+0000\n"
|
||||||
|
"PO-Revision-Date: 2017-06-22 01:11+0000\n"
|
||||||
|
"Last-Translator: Daniel Schweiger <danielcccasle@gmail.com>, 2017\n"
|
||||||
|
"Language-Team: Romanian (https://www.transifex.com/oca/teams/23907/ro/)\n"
|
||||||
|
"Language: ro\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=3; plural=(n==1?0:(((n%100>19)||((n%100==0)&&(n!=0)))?"
|
||||||
|
"2:1));\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Ati primit un email pentru activarea acestui cont!"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr ""
|
||||||
|
"Sa produs o eroare, va rugam sa incercati mai tarziu sau contacteaza-ne."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Nu ati introdus o directie email corespunzatoare."
|
||||||
|
|
@ -0,0 +1,46 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Matjaž Mozetič <m.mozetic@matmoz.si>, 2015
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: server-tools (9.0)\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2016-06-09 12:31+0000\n"
|
||||||
|
"PO-Revision-Date: 2016-05-31 14:47+0000\n"
|
||||||
|
"Last-Translator: OCA Transbot <transbot@odoo-community.org>\n"
|
||||||
|
"Language-Team: Slovenian (http://www.transifex.com/oca/OCA-server-tools-9-0/"
|
||||||
|
"language/sl/)\n"
|
||||||
|
"Language: sl\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=4; plural=(n%100==1 ? 0 : n%100==2 ? 1 : n%100==3 || n"
|
||||||
|
"%100==4 ? 2 : 3);\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Preverite svojo e-pošto za aktiviranje računa!"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr ""
|
||||||
|
"Nekaj je narobe. Ponovno poskusite kasneje ali pa stopite v stik z nami."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Kaže, da to ni e-poštni naslov."
|
||||||
|
|
@ -0,0 +1,44 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * auth_signup_verify_email
|
||||||
|
#
|
||||||
|
# Translators:
|
||||||
|
# Ahmet Altınışık <aaltinisik@altinkaya.com.tr>, 2016
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: server-tools (9.0)\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2016-06-09 12:31+0000\n"
|
||||||
|
"PO-Revision-Date: 2016-05-31 14:47+0000\n"
|
||||||
|
"Last-Translator: OCA Transbot <transbot@odoo-community.org>\n"
|
||||||
|
"Language-Team: Turkish (http://www.transifex.com/oca/OCA-server-tools-9-0/"
|
||||||
|
"language/tr/)\n"
|
||||||
|
"Language: tr\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: nplurals=2; plural=(n > 1);\n"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Another user is already registered using this email address."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Check your email to activate your account!"
|
||||||
|
msgstr "Hesabınızı aktive etmek için e-postanızı kontrol edin!"
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "Something went wrong, please try again later or contact us."
|
||||||
|
msgstr "Bişeyler ters gitti. Lütfen sonra tekrar deneyin ya da bize ulaşın."
|
||||||
|
|
||||||
|
#. module: auth_signup_verify_email
|
||||||
|
#: code:addons/auth_signup_verify_email/controllers/main.py:0
|
||||||
|
#, python-format
|
||||||
|
msgid "That does not seem to be an email address."
|
||||||
|
msgstr "Bu bir e-posta adresi gözükmüyor."
|
||||||
|
|
@ -0,0 +1,6 @@
|
||||||
|
To configure this module, you need to:
|
||||||
|
|
||||||
|
* `Properly configure your outgoing email server(s)
|
||||||
|
<https://www.odoo.com/forum/help-1/question/how-to-configure-email-gateway-282#answer_290>`_.
|
||||||
|
* Go to *Settings > General Settings -> General settings*, search for
|
||||||
|
the *Users* section and enable *Free sign up* in *Customer account*.
|
||||||
|
|
@ -0,0 +1,5 @@
|
||||||
|
* Rafael Blasco <rafaelbn@antiun.com>
|
||||||
|
* Jairo Llopis <yajo.sk8@gmail.com>
|
||||||
|
* Simone Orsi <simone.orsi@camptocamp.com>
|
||||||
|
* Alexandre Díaz <alexandre.diaz@tecnativa.com>
|
||||||
|
* Eugene Molotov <molotov@it-projects.info>
|
||||||
|
|
@ -0,0 +1,5 @@
|
||||||
|
This module extends the functionality of public sign up to force users to
|
||||||
|
provide a valid email address.
|
||||||
|
|
||||||
|
To achieve this, users are not required to provide a password at
|
||||||
|
sign up: they are asked for only at first login attempt.
|
||||||
|
|
@ -0,0 +1,2 @@
|
||||||
|
* Install `email_validator <https://pypi.org/project/email-validator/>`_
|
||||||
|
with ``pip install email_validator`` or equivalent.
|
||||||
|
|
@ -0,0 +1,4 @@
|
||||||
|
To use this module, you need to:
|
||||||
|
|
||||||
|
* Log out.
|
||||||
|
* `Sign up </web/signup>`_ with a valid email.
|
||||||
Binary file not shown.
|
After Width: | Height: | Size: 10 KiB |
File diff suppressed because one or more lines are too long
|
After Width: | Height: | Size: 16 KiB |
|
|
@ -0,0 +1,454 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8" ?>
|
||||||
|
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||||
|
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
|
||||||
|
<head>
|
||||||
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
||||||
|
<meta name="generator" content="Docutils 0.15.1: http://docutils.sourceforge.net/" />
|
||||||
|
<title>Verify email at signup</title>
|
||||||
|
<style type="text/css">
|
||||||
|
|
||||||
|
/*
|
||||||
|
:Author: David Goodger (goodger@python.org)
|
||||||
|
:Id: $Id: html4css1.css 7952 2016-07-26 18:15:59Z milde $
|
||||||
|
:Copyright: This stylesheet has been placed in the public domain.
|
||||||
|
|
||||||
|
Default cascading style sheet for the HTML output of Docutils.
|
||||||
|
|
||||||
|
See http://docutils.sf.net/docs/howto/html-stylesheets.html for how to
|
||||||
|
customize this style sheet.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* used to remove borders from tables and images */
|
||||||
|
.borderless, table.borderless td, table.borderless th {
|
||||||
|
border: 0 }
|
||||||
|
|
||||||
|
table.borderless td, table.borderless th {
|
||||||
|
/* Override padding for "table.docutils td" with "! important".
|
||||||
|
The right padding separates the table cells. */
|
||||||
|
padding: 0 0.5em 0 0 ! important }
|
||||||
|
|
||||||
|
.first {
|
||||||
|
/* Override more specific margin styles with "! important". */
|
||||||
|
margin-top: 0 ! important }
|
||||||
|
|
||||||
|
.last, .with-subtitle {
|
||||||
|
margin-bottom: 0 ! important }
|
||||||
|
|
||||||
|
.hidden {
|
||||||
|
display: none }
|
||||||
|
|
||||||
|
.subscript {
|
||||||
|
vertical-align: sub;
|
||||||
|
font-size: smaller }
|
||||||
|
|
||||||
|
.superscript {
|
||||||
|
vertical-align: super;
|
||||||
|
font-size: smaller }
|
||||||
|
|
||||||
|
a.toc-backref {
|
||||||
|
text-decoration: none ;
|
||||||
|
color: black }
|
||||||
|
|
||||||
|
blockquote.epigraph {
|
||||||
|
margin: 2em 5em ; }
|
||||||
|
|
||||||
|
dl.docutils dd {
|
||||||
|
margin-bottom: 0.5em }
|
||||||
|
|
||||||
|
object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Uncomment (and remove this text!) to get bold-faced definition list terms
|
||||||
|
dl.docutils dt {
|
||||||
|
font-weight: bold }
|
||||||
|
*/
|
||||||
|
|
||||||
|
div.abstract {
|
||||||
|
margin: 2em 5em }
|
||||||
|
|
||||||
|
div.abstract p.topic-title {
|
||||||
|
font-weight: bold ;
|
||||||
|
text-align: center }
|
||||||
|
|
||||||
|
div.admonition, div.attention, div.caution, div.danger, div.error,
|
||||||
|
div.hint, div.important, div.note, div.tip, div.warning {
|
||||||
|
margin: 2em ;
|
||||||
|
border: medium outset ;
|
||||||
|
padding: 1em }
|
||||||
|
|
||||||
|
div.admonition p.admonition-title, div.hint p.admonition-title,
|
||||||
|
div.important p.admonition-title, div.note p.admonition-title,
|
||||||
|
div.tip p.admonition-title {
|
||||||
|
font-weight: bold ;
|
||||||
|
font-family: sans-serif }
|
||||||
|
|
||||||
|
div.attention p.admonition-title, div.caution p.admonition-title,
|
||||||
|
div.danger p.admonition-title, div.error p.admonition-title,
|
||||||
|
div.warning p.admonition-title, .code .error {
|
||||||
|
color: red ;
|
||||||
|
font-weight: bold ;
|
||||||
|
font-family: sans-serif }
|
||||||
|
|
||||||
|
/* Uncomment (and remove this text!) to get reduced vertical space in
|
||||||
|
compound paragraphs.
|
||||||
|
div.compound .compound-first, div.compound .compound-middle {
|
||||||
|
margin-bottom: 0.5em }
|
||||||
|
|
||||||
|
div.compound .compound-last, div.compound .compound-middle {
|
||||||
|
margin-top: 0.5em }
|
||||||
|
*/
|
||||||
|
|
||||||
|
div.dedication {
|
||||||
|
margin: 2em 5em ;
|
||||||
|
text-align: center ;
|
||||||
|
font-style: italic }
|
||||||
|
|
||||||
|
div.dedication p.topic-title {
|
||||||
|
font-weight: bold ;
|
||||||
|
font-style: normal }
|
||||||
|
|
||||||
|
div.figure {
|
||||||
|
margin-left: 2em ;
|
||||||
|
margin-right: 2em }
|
||||||
|
|
||||||
|
div.footer, div.header {
|
||||||
|
clear: both;
|
||||||
|
font-size: smaller }
|
||||||
|
|
||||||
|
div.line-block {
|
||||||
|
display: block ;
|
||||||
|
margin-top: 1em ;
|
||||||
|
margin-bottom: 1em }
|
||||||
|
|
||||||
|
div.line-block div.line-block {
|
||||||
|
margin-top: 0 ;
|
||||||
|
margin-bottom: 0 ;
|
||||||
|
margin-left: 1.5em }
|
||||||
|
|
||||||
|
div.sidebar {
|
||||||
|
margin: 0 0 0.5em 1em ;
|
||||||
|
border: medium outset ;
|
||||||
|
padding: 1em ;
|
||||||
|
background-color: #ffffee ;
|
||||||
|
width: 40% ;
|
||||||
|
float: right ;
|
||||||
|
clear: right }
|
||||||
|
|
||||||
|
div.sidebar p.rubric {
|
||||||
|
font-family: sans-serif ;
|
||||||
|
font-size: medium }
|
||||||
|
|
||||||
|
div.system-messages {
|
||||||
|
margin: 5em }
|
||||||
|
|
||||||
|
div.system-messages h1 {
|
||||||
|
color: red }
|
||||||
|
|
||||||
|
div.system-message {
|
||||||
|
border: medium outset ;
|
||||||
|
padding: 1em }
|
||||||
|
|
||||||
|
div.system-message p.system-message-title {
|
||||||
|
color: red ;
|
||||||
|
font-weight: bold }
|
||||||
|
|
||||||
|
div.topic {
|
||||||
|
margin: 2em }
|
||||||
|
|
||||||
|
h1.section-subtitle, h2.section-subtitle, h3.section-subtitle,
|
||||||
|
h4.section-subtitle, h5.section-subtitle, h6.section-subtitle {
|
||||||
|
margin-top: 0.4em }
|
||||||
|
|
||||||
|
h1.title {
|
||||||
|
text-align: center }
|
||||||
|
|
||||||
|
h2.subtitle {
|
||||||
|
text-align: center }
|
||||||
|
|
||||||
|
hr.docutils {
|
||||||
|
width: 75% }
|
||||||
|
|
||||||
|
img.align-left, .figure.align-left, object.align-left, table.align-left {
|
||||||
|
clear: left ;
|
||||||
|
float: left ;
|
||||||
|
margin-right: 1em }
|
||||||
|
|
||||||
|
img.align-right, .figure.align-right, object.align-right, table.align-right {
|
||||||
|
clear: right ;
|
||||||
|
float: right ;
|
||||||
|
margin-left: 1em }
|
||||||
|
|
||||||
|
img.align-center, .figure.align-center, object.align-center {
|
||||||
|
display: block;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.align-center {
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.align-left {
|
||||||
|
text-align: left }
|
||||||
|
|
||||||
|
.align-center {
|
||||||
|
clear: both ;
|
||||||
|
text-align: center }
|
||||||
|
|
||||||
|
.align-right {
|
||||||
|
text-align: right }
|
||||||
|
|
||||||
|
/* reset inner alignment in figures */
|
||||||
|
div.align-right {
|
||||||
|
text-align: inherit }
|
||||||
|
|
||||||
|
/* div.align-center * { */
|
||||||
|
/* text-align: left } */
|
||||||
|
|
||||||
|
.align-top {
|
||||||
|
vertical-align: top }
|
||||||
|
|
||||||
|
.align-middle {
|
||||||
|
vertical-align: middle }
|
||||||
|
|
||||||
|
.align-bottom {
|
||||||
|
vertical-align: bottom }
|
||||||
|
|
||||||
|
ol.simple, ul.simple {
|
||||||
|
margin-bottom: 1em }
|
||||||
|
|
||||||
|
ol.arabic {
|
||||||
|
list-style: decimal }
|
||||||
|
|
||||||
|
ol.loweralpha {
|
||||||
|
list-style: lower-alpha }
|
||||||
|
|
||||||
|
ol.upperalpha {
|
||||||
|
list-style: upper-alpha }
|
||||||
|
|
||||||
|
ol.lowerroman {
|
||||||
|
list-style: lower-roman }
|
||||||
|
|
||||||
|
ol.upperroman {
|
||||||
|
list-style: upper-roman }
|
||||||
|
|
||||||
|
p.attribution {
|
||||||
|
text-align: right ;
|
||||||
|
margin-left: 50% }
|
||||||
|
|
||||||
|
p.caption {
|
||||||
|
font-style: italic }
|
||||||
|
|
||||||
|
p.credits {
|
||||||
|
font-style: italic ;
|
||||||
|
font-size: smaller }
|
||||||
|
|
||||||
|
p.label {
|
||||||
|
white-space: nowrap }
|
||||||
|
|
||||||
|
p.rubric {
|
||||||
|
font-weight: bold ;
|
||||||
|
font-size: larger ;
|
||||||
|
color: maroon ;
|
||||||
|
text-align: center }
|
||||||
|
|
||||||
|
p.sidebar-title {
|
||||||
|
font-family: sans-serif ;
|
||||||
|
font-weight: bold ;
|
||||||
|
font-size: larger }
|
||||||
|
|
||||||
|
p.sidebar-subtitle {
|
||||||
|
font-family: sans-serif ;
|
||||||
|
font-weight: bold }
|
||||||
|
|
||||||
|
p.topic-title {
|
||||||
|
font-weight: bold }
|
||||||
|
|
||||||
|
pre.address {
|
||||||
|
margin-bottom: 0 ;
|
||||||
|
margin-top: 0 ;
|
||||||
|
font: inherit }
|
||||||
|
|
||||||
|
pre.literal-block, pre.doctest-block, pre.math, pre.code {
|
||||||
|
margin-left: 2em ;
|
||||||
|
margin-right: 2em }
|
||||||
|
|
||||||
|
pre.code .ln { color: grey; } /* line numbers */
|
||||||
|
pre.code, code { background-color: #eeeeee }
|
||||||
|
pre.code .comment, code .comment { color: #5C6576 }
|
||||||
|
pre.code .keyword, code .keyword { color: #3B0D06; font-weight: bold }
|
||||||
|
pre.code .literal.string, code .literal.string { color: #0C5404 }
|
||||||
|
pre.code .name.builtin, code .name.builtin { color: #352B84 }
|
||||||
|
pre.code .deleted, code .deleted { background-color: #DEB0A1}
|
||||||
|
pre.code .inserted, code .inserted { background-color: #A3D289}
|
||||||
|
|
||||||
|
span.classifier {
|
||||||
|
font-family: sans-serif ;
|
||||||
|
font-style: oblique }
|
||||||
|
|
||||||
|
span.classifier-delimiter {
|
||||||
|
font-family: sans-serif ;
|
||||||
|
font-weight: bold }
|
||||||
|
|
||||||
|
span.interpreted {
|
||||||
|
font-family: sans-serif }
|
||||||
|
|
||||||
|
span.option {
|
||||||
|
white-space: nowrap }
|
||||||
|
|
||||||
|
span.pre {
|
||||||
|
white-space: pre }
|
||||||
|
|
||||||
|
span.problematic {
|
||||||
|
color: red }
|
||||||
|
|
||||||
|
span.section-subtitle {
|
||||||
|
/* font-size relative to parent (h1..h6 element) */
|
||||||
|
font-size: 80% }
|
||||||
|
|
||||||
|
table.citation {
|
||||||
|
border-left: solid 1px gray;
|
||||||
|
margin-left: 1px }
|
||||||
|
|
||||||
|
table.docinfo {
|
||||||
|
margin: 2em 4em }
|
||||||
|
|
||||||
|
table.docutils {
|
||||||
|
margin-top: 0.5em ;
|
||||||
|
margin-bottom: 0.5em }
|
||||||
|
|
||||||
|
table.footnote {
|
||||||
|
border-left: solid 1px black;
|
||||||
|
margin-left: 1px }
|
||||||
|
|
||||||
|
table.docutils td, table.docutils th,
|
||||||
|
table.docinfo td, table.docinfo th {
|
||||||
|
padding-left: 0.5em ;
|
||||||
|
padding-right: 0.5em ;
|
||||||
|
vertical-align: top }
|
||||||
|
|
||||||
|
table.docutils th.field-name, table.docinfo th.docinfo-name {
|
||||||
|
font-weight: bold ;
|
||||||
|
text-align: left ;
|
||||||
|
white-space: nowrap ;
|
||||||
|
padding-left: 0 }
|
||||||
|
|
||||||
|
/* "booktabs" style (no vertical lines) */
|
||||||
|
table.docutils.booktabs {
|
||||||
|
border: 0px;
|
||||||
|
border-top: 2px solid;
|
||||||
|
border-bottom: 2px solid;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
table.docutils.booktabs * {
|
||||||
|
border: 0px;
|
||||||
|
}
|
||||||
|
table.docutils.booktabs th {
|
||||||
|
border-bottom: thin solid;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 tt.docutils, h2 tt.docutils, h3 tt.docutils,
|
||||||
|
h4 tt.docutils, h5 tt.docutils, h6 tt.docutils {
|
||||||
|
font-size: 100% }
|
||||||
|
|
||||||
|
ul.auto-toc {
|
||||||
|
list-style-type: none }
|
||||||
|
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="document" id="verify-email-at-signup">
|
||||||
|
<h1 class="title">Verify email at signup</h1>
|
||||||
|
|
||||||
|
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
|
!! This file is generated by oca-gen-addon-readme !!
|
||||||
|
!! changes will be overwritten. !!
|
||||||
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! -->
|
||||||
|
<p><a class="reference external" href="https://odoo-community.org/page/development-status"><img alt="Beta" src="https://img.shields.io/badge/maturity-Beta-yellow.png" /></a> <a class="reference external" href="http://www.gnu.org/licenses/agpl-3.0-standalone.html"><img alt="License: AGPL-3" src="https://img.shields.io/badge/licence-AGPL--3-blue.png" /></a> <a class="reference external" href="https://github.com/OCA/server-auth/tree/13.0/auth_signup_verify_email"><img alt="OCA/server-auth" src="https://img.shields.io/badge/github-OCA%2Fserver--auth-lightgray.png?logo=github" /></a> <a class="reference external" href="https://translation.odoo-community.org/projects/server-auth-13-0/server-auth-13-0-auth_signup_verify_email"><img alt="Translate me on Weblate" src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png" /></a> <a class="reference external" href="https://runbot.odoo-community.org/runbot/251/13.0"><img alt="Try me on Runbot" src="https://img.shields.io/badge/runbot-Try%20me-875A7B.png" /></a></p>
|
||||||
|
<p>This module extends the functionality of public sign up to force users to
|
||||||
|
provide a valid email address.</p>
|
||||||
|
<p>To achieve this, users are not required to provide a password at
|
||||||
|
sign up: they are asked for only at first login attempt.</p>
|
||||||
|
<p><strong>Table of contents</strong></p>
|
||||||
|
<div class="contents local topic" id="contents">
|
||||||
|
<ul class="simple">
|
||||||
|
<li><a class="reference internal" href="#installation" id="id1">Installation</a></li>
|
||||||
|
<li><a class="reference internal" href="#configuration" id="id2">Configuration</a></li>
|
||||||
|
<li><a class="reference internal" href="#usage" id="id3">Usage</a></li>
|
||||||
|
<li><a class="reference internal" href="#bug-tracker" id="id4">Bug Tracker</a></li>
|
||||||
|
<li><a class="reference internal" href="#credits" id="id5">Credits</a><ul>
|
||||||
|
<li><a class="reference internal" href="#authors" id="id6">Authors</a></li>
|
||||||
|
<li><a class="reference internal" href="#contributors" id="id7">Contributors</a></li>
|
||||||
|
<li><a class="reference internal" href="#maintainers" id="id8">Maintainers</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="installation">
|
||||||
|
<h1><a class="toc-backref" href="#id1">Installation</a></h1>
|
||||||
|
<ul class="simple">
|
||||||
|
<li>Install <a class="reference external" href="https://pypi.org/project/email-validator/">email_validator</a>
|
||||||
|
with <tt class="docutils literal">pip install email_validator</tt> or equivalent.</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="configuration">
|
||||||
|
<h1><a class="toc-backref" href="#id2">Configuration</a></h1>
|
||||||
|
<p>To configure this module, you need to:</p>
|
||||||
|
<ul class="simple">
|
||||||
|
<li><a class="reference external" href="https://www.odoo.com/forum/help-1/question/how-to-configure-email-gateway-282#answer_290">Properly configure your outgoing email server(s)</a>.</li>
|
||||||
|
<li>Go to <em>Settings > General Settings -> General settings</em>, search for
|
||||||
|
the <em>Users</em> section and enable <em>Free sign up</em> in <em>Customer account</em>.</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="usage">
|
||||||
|
<h1><a class="toc-backref" href="#id3">Usage</a></h1>
|
||||||
|
<p>To use this module, you need to:</p>
|
||||||
|
<ul class="simple">
|
||||||
|
<li>Log out.</li>
|
||||||
|
<li><a class="reference external" href="/web/signup">Sign up</a> with a valid email.</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="bug-tracker">
|
||||||
|
<h1><a class="toc-backref" href="#id4">Bug Tracker</a></h1>
|
||||||
|
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/server-auth/issues">GitHub Issues</a>.
|
||||||
|
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
|
||||||
|
<a class="reference external" href="https://github.com/OCA/server-auth/issues/new?body=module:%20auth_signup_verify_email%0Aversion:%2013.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**">feedback</a>.</p>
|
||||||
|
<p>Do not contact contributors directly about support or help with technical issues.</p>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="credits">
|
||||||
|
<h1><a class="toc-backref" href="#id5">Credits</a></h1>
|
||||||
|
<div class="section" id="authors">
|
||||||
|
<h2><a class="toc-backref" href="#id6">Authors</a></h2>
|
||||||
|
<ul class="simple">
|
||||||
|
<li>Antiun Ingeniería S.L.</li>
|
||||||
|
<li>Tecnativa</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="contributors">
|
||||||
|
<h2><a class="toc-backref" href="#id7">Contributors</a></h2>
|
||||||
|
<ul class="simple">
|
||||||
|
<li>Rafael Blasco <<a class="reference external" href="mailto:rafaelbn@antiun.com">rafaelbn@antiun.com</a>></li>
|
||||||
|
<li>Jairo Llopis <<a class="reference external" href="mailto:yajo.sk8@gmail.com">yajo.sk8@gmail.com</a>></li>
|
||||||
|
<li>Simone Orsi <<a class="reference external" href="mailto:simone.orsi@camptocamp.com">simone.orsi@camptocamp.com</a>></li>
|
||||||
|
<li>Alexandre Díaz <<a class="reference external" href="mailto:alexandre.diaz@tecnativa.com">alexandre.diaz@tecnativa.com</a>></li>
|
||||||
|
<li>Eugene Molotov <<a class="reference external" href="mailto:molotov@it-projects.info">molotov@it-projects.info</a>></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="section" id="maintainers">
|
||||||
|
<h2><a class="toc-backref" href="#id8">Maintainers</a></h2>
|
||||||
|
<p>This module is maintained by the OCA.</p>
|
||||||
|
<a class="reference external image-reference" href="https://odoo-community.org"><img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" /></a>
|
||||||
|
<p>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.</p>
|
||||||
|
<p>This module is part of the <a class="reference external" href="https://github.com/OCA/server-auth/tree/13.0/auth_signup_verify_email">OCA/server-auth</a> project on GitHub.</p>
|
||||||
|
<p>You are welcome to contribute. To learn how please visit <a class="reference external" href="https://odoo-community.org/page/Contribute">https://odoo-community.org/page/Contribute</a>.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
|
@ -0,0 +1,4 @@
|
||||||
|
# Copyright 2016 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from . import test_verify_email
|
||||||
|
|
@ -0,0 +1,49 @@
|
||||||
|
# Copyright 2016 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from lxml.html import document_fromstring
|
||||||
|
from mock import patch
|
||||||
|
|
||||||
|
from odoo.tests.common import HttpCase
|
||||||
|
from odoo.tools.misc import mute_logger
|
||||||
|
|
||||||
|
from odoo.addons.mail.models import mail_template
|
||||||
|
|
||||||
|
|
||||||
|
class UICase(HttpCase):
|
||||||
|
def setUp(self):
|
||||||
|
super().setUp()
|
||||||
|
if "website" in self.env:
|
||||||
|
# Enable public signup in website if it is installed; otherwise
|
||||||
|
# tests here would fail
|
||||||
|
current_website = self.env["website"].get_current_website()
|
||||||
|
current_website.auth_signup_uninvited = "b2c"
|
||||||
|
self.env["ir.config_parameter"].set_param("auth_signup.invitation_scope", "b2c")
|
||||||
|
self.data = {
|
||||||
|
"csrf_token": self.csrf_token(),
|
||||||
|
"name": "Somebody",
|
||||||
|
}
|
||||||
|
|
||||||
|
def html_doc(self, url="/web/signup", data=None, timeout=30):
|
||||||
|
"""Get an HTML LXML document."""
|
||||||
|
with patch(mail_template.__name__ + ".MailTemplate.send_mail"):
|
||||||
|
resp = self.url_open(url, data=data, timeout=timeout)
|
||||||
|
return document_fromstring(resp.content)
|
||||||
|
|
||||||
|
def csrf_token(self):
|
||||||
|
"""Get a valid CSRF token."""
|
||||||
|
doc = self.html_doc()
|
||||||
|
return doc.xpath("//input[@name='csrf_token']")[0].get("value")
|
||||||
|
|
||||||
|
def test_bad_email(self):
|
||||||
|
"""Test rejection of bad emails."""
|
||||||
|
self.data["login"] = "bad email"
|
||||||
|
doc = self.html_doc(data=self.data)
|
||||||
|
self.assertTrue(doc.xpath('//p[@class="alert alert-danger"]'))
|
||||||
|
|
||||||
|
@mute_logger("odoo.addons.auth_signup_verify_email.controllers.main")
|
||||||
|
def test_good_email(self):
|
||||||
|
"""Test acceptance of good emails."""
|
||||||
|
self.data["login"] = "good@example.com"
|
||||||
|
doc = self.html_doc(data=self.data)
|
||||||
|
self.assertTrue(doc.xpath('//p[@class="alert alert-success"]'))
|
||||||
|
|
@ -0,0 +1,16 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" ?>
|
||||||
|
<!-- Copyright 2016 Jairo Llopis <jairo.llopis@tecnativa.com>
|
||||||
|
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
|
||||||
|
<odoo>
|
||||||
|
<template id="signup_fields" inherit_id="auth_signup.fields">
|
||||||
|
<xpath expr="//div[hasclass('field-password')]" position="attributes">
|
||||||
|
<attribute name="t-if">only_passwords</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//div[hasclass('field-confirm_password')]" position="attributes">
|
||||||
|
<attribute name="t-if">only_passwords</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//input[@name='login']" position="attributes">
|
||||||
|
<attribute name="type">email</attribute>
|
||||||
|
</xpath>
|
||||||
|
</template>
|
||||||
|
</odoo>
|
||||||
|
|
@ -0,0 +1,2 @@
|
||||||
|
from . import models
|
||||||
|
from . import wizards
|
||||||
|
|
@ -0,0 +1,30 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
{
|
||||||
|
'name': "Clients Periodic Backups | SaaS",
|
||||||
|
|
||||||
|
'summary': """
|
||||||
|
Take Periodic Backups of Client Instances""",
|
||||||
|
|
||||||
|
'description': """
|
||||||
|
Take Periodic Backups of Client Instances""",
|
||||||
|
|
||||||
|
'author': "Muhammad Awais",
|
||||||
|
'website': "https://codetuple.io",
|
||||||
|
'category': 'Uncategorized',
|
||||||
|
'version': '2.0.0',
|
||||||
|
|
||||||
|
# any module necessary for this one to work correctly
|
||||||
|
'depends': ['base', 'kk_odoo_saas','queue_job'],
|
||||||
|
|
||||||
|
# always loaded
|
||||||
|
'data': [
|
||||||
|
# "security/security.xml",
|
||||||
|
'security/ir.model.access.csv',
|
||||||
|
'wizards/backup_restore.xml',
|
||||||
|
'views/views.xml',
|
||||||
|
'views/app_views.xml',
|
||||||
|
'data/backup_ignite_cron.xml',
|
||||||
|
],
|
||||||
|
"application": True,
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,14 @@
|
||||||
|
<?xml version="1.0"?>
|
||||||
|
<odoo>
|
||||||
|
<data>
|
||||||
|
<record id="backup_process_ignite_crone" model="ir.cron">
|
||||||
|
<field name="name">SaaS: Backup Process Ignite Cron</field>
|
||||||
|
<field name="model_id" ref="model_kk_odoo_saas_app"/>
|
||||||
|
<field name="state">code</field>
|
||||||
|
<field name="code">model.ignite_backup_server_cron()</field>
|
||||||
|
<field name="interval_number">1</field>
|
||||||
|
<field name="interval_type">days</field>
|
||||||
|
<field name="numbercall">-1</field>
|
||||||
|
</record>
|
||||||
|
</data>
|
||||||
|
</odoo>
|
||||||
|
|
@ -0,0 +1,2 @@
|
||||||
|
from . import saas_app
|
||||||
|
from . import models
|
||||||
|
|
@ -0,0 +1,137 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import os
|
||||||
|
|
||||||
|
from odoo import models, fields, api, _
|
||||||
|
from odoo.exceptions import UserError, MissingError
|
||||||
|
import requests
|
||||||
|
import xmlrpc
|
||||||
|
import logging
|
||||||
|
import base64
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class SaaSAppBackup(models.Model):
|
||||||
|
_name = 'kk_odoo_saas.app.backup'
|
||||||
|
_description = 'SaaS App Backup'
|
||||||
|
|
||||||
|
name = fields.Char()
|
||||||
|
app = fields.Many2one('kk_odoo_saas.app', 'SaaS App')
|
||||||
|
file_name = fields.Char(string="File Name")
|
||||||
|
file_path = fields.Char(string="File Path")
|
||||||
|
url = fields.Char(string="Url")
|
||||||
|
backup_date_time = fields.Datetime(string="Backup Time (UTC)")
|
||||||
|
status = fields.Selection(string="Status", selection=[('failed', 'Failed'), ('success', 'Success')])
|
||||||
|
message = fields.Char(string="Message")
|
||||||
|
file_size = fields.Char(string="File Size")
|
||||||
|
|
||||||
|
def download_db_file(self):
|
||||||
|
"""
|
||||||
|
to download the database backup, it stores the file in attachment
|
||||||
|
:return: Action
|
||||||
|
"""
|
||||||
|
file_path = self.file_path
|
||||||
|
_logger.info("------------ %r ----------------" % file_path)
|
||||||
|
if self.url:
|
||||||
|
return {
|
||||||
|
'type': 'ir.actions.act_url',
|
||||||
|
'url': self.url,
|
||||||
|
'target': 'new',
|
||||||
|
}
|
||||||
|
try:
|
||||||
|
with open(file_path, 'rb') as reader:
|
||||||
|
result = base64.b64encode(reader.read())
|
||||||
|
except IOError as e:
|
||||||
|
raise MissingError('Unable to find File on the path')
|
||||||
|
attachment_obj = self.env['ir.attachment'].sudo()
|
||||||
|
name = self.file_name
|
||||||
|
attachment_id = attachment_obj.create({
|
||||||
|
'name': name,
|
||||||
|
'datas': result,
|
||||||
|
'public': False
|
||||||
|
})
|
||||||
|
download_url = '/web/content/' + str(attachment_id.id) + '?download=true'
|
||||||
|
_logger.info("--- %r ----" % download_url)
|
||||||
|
self.url = download_url
|
||||||
|
return {
|
||||||
|
'type': 'ir.actions.act_url',
|
||||||
|
'url': download_url,
|
||||||
|
'target': 'new',
|
||||||
|
}
|
||||||
|
|
||||||
|
def action_restore_backup_to_instance(self, restore_to_id=False):
|
||||||
|
"""
|
||||||
|
it will restore the backup to a new instance,
|
||||||
|
the instance should be created manually,
|
||||||
|
and there should be no database at new_instance.com/web/database/selector
|
||||||
|
:param: restore_to_id is kk_saa_app object on which we have to restore backup
|
||||||
|
:return: False
|
||||||
|
"""
|
||||||
|
if self.app and restore_to_id:
|
||||||
|
restore_url = restore_to_id.get_url()
|
||||||
|
if self.file_path and os.path.exists(self.file_path) and requests.get(restore_url).status_code < 400:
|
||||||
|
db_list = []
|
||||||
|
try:
|
||||||
|
db_list = xmlrpc.client.ServerProxy(restore_url + '/xmlrpc/db').list()
|
||||||
|
except xmlrpc.client.ProtocolError as e:
|
||||||
|
_logger.info("There is no database on Db selector")
|
||||||
|
|
||||||
|
_logger.info("All Databases on Postgres Server -> {} <-".format(db_list))
|
||||||
|
_logger.info("New Db name: {}".format(restore_to_id.app_name))
|
||||||
|
if restore_to_id.app_name not in db_list:
|
||||||
|
self.restore_backup_to_client(self.file_path, restore_url, restore_to_id.app_name,
|
||||||
|
restore_to_id.backup_master_pass)
|
||||||
|
else:
|
||||||
|
raise UserError("Cant restore Backup, Database already existed, please delete it.")
|
||||||
|
else:
|
||||||
|
raise UserError("Cant restore Backup! the url is not accessible or backup file not exists.")
|
||||||
|
else:
|
||||||
|
_logger.error("Cant restore Backup, Backup Id, or Restore App Missing")
|
||||||
|
raise UserError("Cant restore Backup, Backup Id, or Restore App Missing")
|
||||||
|
|
||||||
|
def restore_backup_to_client(self, file_path, restore_url, db_name, master_pwd):
|
||||||
|
if file_path and restore_url and db_name and master_pwd:
|
||||||
|
restore_url = restore_url + '/web/database/restore'
|
||||||
|
data = {
|
||||||
|
'master_pwd': master_pwd,
|
||||||
|
'name': db_name,
|
||||||
|
'copy': 'true',
|
||||||
|
'backup_file': '@' + file_path
|
||||||
|
}
|
||||||
|
backup = open(file_path, "rb")
|
||||||
|
try:
|
||||||
|
response = requests.post(restore_url, data=data, files={"backup_file": backup})
|
||||||
|
if response.status_code == 200:
|
||||||
|
_logger.info("Restore Done, this is the response Code: {}".format(response.status_code))
|
||||||
|
else:
|
||||||
|
_logger.info("Restore Done, this is the response Code: {}".format(response.status_code))
|
||||||
|
|
||||||
|
return {
|
||||||
|
'success': True,
|
||||||
|
}
|
||||||
|
except Exception as e:
|
||||||
|
return {
|
||||||
|
'success': False,
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
_logger.error("Cant restore Db One of the parameter is Missing")
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def create(self, vals):
|
||||||
|
vals['name'] = self.env['ir.sequence'].next_by_code('saas_app.backup')
|
||||||
|
res = super(SaaSAppBackup, self).create(vals)
|
||||||
|
return res
|
||||||
|
|
||||||
|
def calc_backup_size(self):
|
||||||
|
if not os.path.exists(self.file_path):
|
||||||
|
return
|
||||||
|
# calculate file size in KB, MB, GB
|
||||||
|
|
||||||
|
def convert_bytes(size):
|
||||||
|
""" Convert bytes to KB, or MB or GB"""
|
||||||
|
for x in ['bytes', 'KB', 'MB', 'GB', 'TB']:
|
||||||
|
if size < 1024.0:
|
||||||
|
return "%3.1f %s" % (size, x)
|
||||||
|
size /= 1024.0
|
||||||
|
|
||||||
|
self.file_size = convert_bytes(os.path.getsize(self.file_path))
|
||||||
|
|
@ -0,0 +1,124 @@
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
import requests
|
||||||
|
from odoo import models, fields, api, _
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class SaaSApp(models.Model):
|
||||||
|
_inherit = 'kk_odoo_saas.app'
|
||||||
|
backup_db_name = fields.Char(string="Database Name", default=lambda a: a.client_db_name)
|
||||||
|
backup_master_pass = fields.Char(string="Master Password")
|
||||||
|
backups_enabled = fields.Boolean()
|
||||||
|
|
||||||
|
backups = fields.Many2many(comodel_name='kk_odoo_saas.app.backup', string='Backups')
|
||||||
|
|
||||||
|
def action_create_backup(self):
|
||||||
|
"""
|
||||||
|
It is being called from 2 locations
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
for app in self:
|
||||||
|
response = self.backup_db()
|
||||||
|
backup = self.env['kk_odoo_saas.app.backup'].create({'backup_date_time': fields.Datetime.now(),
|
||||||
|
'app': app.id,
|
||||||
|
'file_name': response.get('filename'),
|
||||||
|
'file_path': response.get('filepath'),
|
||||||
|
'message': response.get('message')
|
||||||
|
})
|
||||||
|
if response.get('success'):
|
||||||
|
backup.write({'status': 'success'})
|
||||||
|
else:
|
||||||
|
backup.write({'status': 'failed'})
|
||||||
|
app.write({'backups': [(4, backup.id)]})
|
||||||
|
|
||||||
|
def action_delete_old_backup(self):
|
||||||
|
for app in self:
|
||||||
|
for backup in app.backups:
|
||||||
|
if backup.backup_date_time < fields.Datetime.now() - timedelta(days=7.0):
|
||||||
|
if os.path.exists(backup.file_path):
|
||||||
|
try:
|
||||||
|
os.remove(backup.file_path)
|
||||||
|
if backup.url:
|
||||||
|
# deleting the attachments related to this backup
|
||||||
|
att_id = backup.url.replace('?download=true', '').replace('/web/content/', '')
|
||||||
|
if att_id:
|
||||||
|
try:
|
||||||
|
attch_id = int(att_id)
|
||||||
|
if attch_id:
|
||||||
|
self.env['ir.attachment'].browse([attch_id]).unlink()
|
||||||
|
except ValueError as e:
|
||||||
|
_logger.error(e)
|
||||||
|
backup.unlink()
|
||||||
|
except OSError as e:
|
||||||
|
_logger.error("Error while deleting file: %s - %s." % (e.filename, e.strerror))
|
||||||
|
else:
|
||||||
|
_logger.error("The file does not exist")
|
||||||
|
|
||||||
|
def backup_db(self):
|
||||||
|
"""
|
||||||
|
Actual Backup function
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
# get the creds for db manager
|
||||||
|
data = {
|
||||||
|
'master_pwd': self.backup_master_pass,
|
||||||
|
'name': self.backup_db_name,
|
||||||
|
'backup_format': 'zip'
|
||||||
|
}
|
||||||
|
|
||||||
|
client_url = 'https://{0}{1}'.format(self.sub_domain_name, self.domain_name)
|
||||||
|
msg = ''
|
||||||
|
|
||||||
|
# where we want to store backups, in the linux user, with which the odoo-service is running
|
||||||
|
backup_dir = os.path.join(os.path.expanduser('~'), 'client_backups')
|
||||||
|
if not os.path.exists(backup_dir):
|
||||||
|
os.mkdir(backup_dir)
|
||||||
|
|
||||||
|
backup_dir = os.path.join(backup_dir, self.sub_domain_name)
|
||||||
|
if not os.path.exists(backup_dir):
|
||||||
|
os.mkdir(backup_dir)
|
||||||
|
|
||||||
|
client_url += '/web/database/backup'
|
||||||
|
# Without Streaming method
|
||||||
|
# response = requests.post(client_url, data=data)
|
||||||
|
# Streaming zip, so that everything is not stored in RAM.
|
||||||
|
|
||||||
|
try:
|
||||||
|
filename = self.backup_db_name + '-' + fields.Datetime.now().strftime("%m-%d-%Y-%H-%M") + '.zip'
|
||||||
|
backed_up_file_path = os.path.join(backup_dir, filename)
|
||||||
|
with requests.post(client_url, data=data, stream=True) as response:
|
||||||
|
response.raise_for_status()
|
||||||
|
with open(os.path.join(backup_dir, filename), 'wb') as file:
|
||||||
|
for chunk in response.iter_content(chunk_size=1024):
|
||||||
|
if chunk:
|
||||||
|
file.write(chunk)
|
||||||
|
msg = 'Database backup Successful at ' + fields.Datetime.now().strftime("%m-%d-%Y-%H:%M:%S")
|
||||||
|
return {
|
||||||
|
'success': True,
|
||||||
|
'msg': msg,
|
||||||
|
'filename': filename,
|
||||||
|
'filepath': backed_up_file_path
|
||||||
|
}
|
||||||
|
except Exception as e:
|
||||||
|
msg = 'Failed at ' + fields.Datetime.now().strftime("%m-%d-%Y-%H:%M:%S") + ' ' + str(e)
|
||||||
|
return {
|
||||||
|
'success': False,
|
||||||
|
'msg': msg
|
||||||
|
}
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def ignite_backup_server_cron(self):
|
||||||
|
"""
|
||||||
|
A Scheduled Action which will take new backups and del old
|
||||||
|
:return: False
|
||||||
|
"""
|
||||||
|
# search for saas instance in lanched and modified states and backups enabled
|
||||||
|
apps = self.env['kk_odoo_saas.app'].sudo().search(
|
||||||
|
[('status', 'in', ['l', 'm']), ('backups_enabled', '=', True)])
|
||||||
|
|
||||||
|
for app in apps:
|
||||||
|
app.action_create_backup()
|
||||||
|
app.action_delete_old_backup()
|
||||||
|
|
@ -0,0 +1,3 @@
|
||||||
|
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
||||||
|
access_saas_app_backups,access_saas_app_backups,model_kk_odoo_saas_app_backup,kk_odoo_saas.group_saas_manager,1,1,1,1
|
||||||
|
access_saas_app_backup_restore_wizard,SaaS Manager Backup Restore Permissions,model_saas_client_backup_restore_wizard,kk_odoo_saas.group_saas_manager,1,1,1,1
|
||||||
|
Binary file not shown.
|
After Width: | Height: | Size: 4.9 KiB |
|
|
@ -0,0 +1,37 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<odoo>
|
||||||
|
<data>
|
||||||
|
<!-- Inherit Form View to Modify it -->
|
||||||
|
<record id="saas_app_form_backup_inherit" model="ir.ui.view">
|
||||||
|
<field name="name">SaaS App inherit for Backups</field>
|
||||||
|
<field name="model">kk_odoo_saas.app</field>
|
||||||
|
<field name="inherit_id" ref="kk_odoo_saas.kk_odoo_saas_app_view_form"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//page[@name='k8s_logs']" position="after">
|
||||||
|
<page name="backups" string="Backups">
|
||||||
|
<group>
|
||||||
|
<group>
|
||||||
|
<field name="backups_enabled" widget="boolean_toggle"/>
|
||||||
|
<field name="backup_db_name"
|
||||||
|
attrs="{'required': [('backups_enabled', '=', True)], 'invisible': [('backups_enabled', '!=', True)]}"/>
|
||||||
|
</group>
|
||||||
|
<group>
|
||||||
|
<field name="backup_master_pass"
|
||||||
|
attrs="{'required': [('backups_enabled', '=', True)], 'invisible': [('backups_enabled', '!=', True)]}"
|
||||||
|
password="1"/>
|
||||||
|
<button name="action_create_backup"
|
||||||
|
type="object"
|
||||||
|
string="Take Backup"
|
||||||
|
attrs="{'invisible': ['|', ('status','in',('d','del')),('backups_enabled', '=', False)]}"
|
||||||
|
/>
|
||||||
|
</group>
|
||||||
|
</group>
|
||||||
|
<group>
|
||||||
|
<field name="backups" nolabel="1" readonly="1"/>
|
||||||
|
</group>
|
||||||
|
</page>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
</data>
|
||||||
|
</odoo>
|
||||||
|
|
@ -0,0 +1,75 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<odoo>
|
||||||
|
<data>
|
||||||
|
<record id="kk_odoo_saas_app_backup_form_view" model="ir.ui.view">
|
||||||
|
<field name="name">kk_odoo_saas.kk_odoo_saas.app.backup.form</field>
|
||||||
|
<field name="model">kk_odoo_saas.app.backup</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form>
|
||||||
|
<header>
|
||||||
|
<button name="download_db_file"
|
||||||
|
type="object"
|
||||||
|
string="Download File"
|
||||||
|
/>
|
||||||
|
<button name="action_restore_backup_to_instance"
|
||||||
|
type="object"
|
||||||
|
string="Restore to Instance"
|
||||||
|
/>
|
||||||
|
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<sheet>
|
||||||
|
<div class="oe_title">
|
||||||
|
<label for="name"/>
|
||||||
|
<h1>
|
||||||
|
<field name="name" placeholder="Title" readonly="1"/>
|
||||||
|
</h1>
|
||||||
|
</div>
|
||||||
|
<group>
|
||||||
|
<group>
|
||||||
|
<field name="app"/>
|
||||||
|
<field name="file_name"/>
|
||||||
|
<field name="file_path"/>
|
||||||
|
</group>
|
||||||
|
<group>
|
||||||
|
<field name="url"/>
|
||||||
|
<field name="backup_date_time"/>
|
||||||
|
<field name="status"/>
|
||||||
|
<field name="message"/>
|
||||||
|
</group>
|
||||||
|
</group>
|
||||||
|
</sheet>
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="kk_odoo_saas_app_backup_tree_view" model="ir.ui.view">
|
||||||
|
<field name="name">kk_odoo_saas.kk_odoo_saas.app.backup.tree</field>
|
||||||
|
<field name="model">kk_odoo_saas.app.backup</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<tree string="App Backup" decoration-success="status=='success'" decoration-danger="status!='success'">
|
||||||
|
<field name="name" readonly="1"/>
|
||||||
|
<field name="file_name"/>
|
||||||
|
<field name="backup_date_time"/>
|
||||||
|
<field name="status"/>
|
||||||
|
<field name="file_size"/>
|
||||||
|
<button name="download_db_file"
|
||||||
|
type="object" icon="fa-cloud-download"
|
||||||
|
string="Download Zip File"
|
||||||
|
class="btn-secondary"
|
||||||
|
/>
|
||||||
|
<button name="ct_client_backup.action_saas_client_backup_restore_wizard"
|
||||||
|
string="Restore to Instance" type="action" icon="fa-cloud-upload"
|
||||||
|
class="btn-secondary"
|
||||||
|
context="{'default_backup_id':id}"
|
||||||
|
attrs="{'invisible':[('status','=', 'failed')]}" />
|
||||||
|
|
||||||
|
<button name="calc_backup_size" type="object"
|
||||||
|
icon="fa-refresh" class="btn-secondary"
|
||||||
|
string="Calculate file Size"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
</data>
|
||||||
|
</odoo>
|
||||||
|
|
@ -0,0 +1 @@
|
||||||
|
from . import backup_restore
|
||||||
|
|
@ -0,0 +1,23 @@
|
||||||
|
from odoo import fields, models
|
||||||
|
import logging
|
||||||
|
from odoo.exceptions import UserError, MissingError
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class BackupRestore(models.TransientModel):
|
||||||
|
_name = 'saas.client.backup.restore.wizard'
|
||||||
|
name = fields.Char('Name')
|
||||||
|
backup_id = fields.Many2one('kk_odoo_saas.app.backup', 'Backup Name')
|
||||||
|
restore_to = fields.Many2one('kk_odoo_saas.app', 'Restore Backup To')
|
||||||
|
|
||||||
|
def action_call_restore_function(self):
|
||||||
|
"""
|
||||||
|
It will call the Backup Function Async, Thanks to queue_job module
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
if self.backup_id and self.backup_id.app and self.restore_to:
|
||||||
|
self.backup_id.action_restore_backup_to_instance(self.restore_to)
|
||||||
|
else:
|
||||||
|
_logger.error("Cant restore Backup, Backup Id, or Restore App Missing")
|
||||||
|
raise UserError("Cant restore Backup, Backup Id, or Restore App Missing")
|
||||||
|
|
@ -0,0 +1,45 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<odoo>
|
||||||
|
<data>
|
||||||
|
<record id="saas_client_backup_restore_view_form" model="ir.ui.view">
|
||||||
|
<field name="name">saas_client_backup_restore_view_form</field>
|
||||||
|
<field name="model">saas.client.backup.restore.wizard</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form>
|
||||||
|
<sheet>
|
||||||
|
<group>
|
||||||
|
<group>
|
||||||
|
<field name="name"/>
|
||||||
|
<field name="backup_id" required="1"/>
|
||||||
|
</group>
|
||||||
|
<group>
|
||||||
|
<field name="restore_to" string="Restore to SaaS Instance" required="1"/>
|
||||||
|
</group>
|
||||||
|
</group>
|
||||||
|
</sheet>
|
||||||
|
<footer>
|
||||||
|
<button
|
||||||
|
name="action_call_restore_function"
|
||||||
|
string="Start Restoring Process"
|
||||||
|
type="object"
|
||||||
|
class="oe_highlight"
|
||||||
|
/>
|
||||||
|
<button string="Cancel" class="oe_link" special="cancel"/>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="action_saas_client_backup_restore_wizard" model="ir.actions.act_window">
|
||||||
|
<field name="name">Restore the Backup to SaaS App</field>
|
||||||
|
<field name="res_model">saas.client.backup.restore.wizard</field>
|
||||||
|
<field name="view_mode">form</field>
|
||||||
|
<field name="view_id" ref="saas_client_backup_restore_view_form"/>
|
||||||
|
<field name="target">new</field>
|
||||||
|
<field name="binding_model_id" ref="ct_client_backup.model_kk_odoo_saas_app_backup"/>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
|
||||||
|
</data>
|
||||||
|
</odoo>
|
||||||
Loading…
Reference in New Issue