[ADD] dependant modules

This commit is contained in:
Muhammad 2024-04-07 12:46:04 +05:00
parent fa3d921e2d
commit e8bc408a7a
42 changed files with 1952 additions and 0 deletions

View File

@ -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.

View File

@ -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

View File

@ -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,
}

View File

@ -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

View File

@ -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)

View File

@ -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 "لا يبدو أن هذا عنوان بريد إلكتروني."

View File

@ -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 ""

View File

@ -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."

View File

@ -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."

View File

@ -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."

View File

@ -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."

View File

@ -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."

View File

@ -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."

View File

@ -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"

View File

@ -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."

View File

@ -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."

View File

@ -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."

View File

@ -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*.

View File

@ -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>

View File

@ -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.

View File

@ -0,0 +1,2 @@
* Install `email_validator <https://pypi.org/project/email-validator/>`_
with ``pip install email_validator`` or equivalent.

View File

@ -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

View File

@ -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 &gt; General Settings -&gt; 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 &lt;<a class="reference external" href="mailto:rafaelbn&#64;antiun.com">rafaelbn&#64;antiun.com</a>&gt;</li>
<li>Jairo Llopis &lt;<a class="reference external" href="mailto:yajo.sk8&#64;gmail.com">yajo.sk8&#64;gmail.com</a>&gt;</li>
<li>Simone Orsi &lt;<a class="reference external" href="mailto:simone.orsi&#64;camptocamp.com">simone.orsi&#64;camptocamp.com</a>&gt;</li>
<li>Alexandre Díaz &lt;<a class="reference external" href="mailto:alexandre.diaz&#64;tecnativa.com">alexandre.diaz&#64;tecnativa.com</a>&gt;</li>
<li>Eugene Molotov &lt;<a class="reference external" href="mailto:molotov&#64;it-projects.info">molotov&#64;it-projects.info</a>&gt;</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>

View File

@ -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

View File

@ -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"]'))

View File

@ -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>

2
ct_client_backup/__init__.py Executable file
View File

@ -0,0 +1,2 @@
from . import models
from . import wizards

View File

@ -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,
}

View File

View File

@ -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>

View File

@ -0,0 +1,2 @@
from . import saas_app
from . import models

View File

@ -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))

View File

@ -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()

View File

@ -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
1 id name model_id:id group_id:id perm_read perm_write perm_create perm_unlink
2 access_saas_app_backups access_saas_app_backups model_kk_odoo_saas_app_backup kk_odoo_saas.group_saas_manager 1 1 1 1
3 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

View File

@ -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>

View File

@ -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>

View File

@ -0,0 +1 @@
from . import backup_restore

View File

@ -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")

View File

@ -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>