# -*- coding: utf-8 -*- ############################################################################### # # Cybrosys Technologies Pvt. Ltd. # # Copyright (C) 2024-TODAY Cybrosys Technologies() # Author: Cybrosys Techno Solutions (odoo@cybrosys.com) # # You can modify it under the terms of the GNU AFFERO # GENERAL PUBLIC LICENSE (AGPL v3), Version 3. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. # # You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE # (AGPL v3) along with this program. # If not, see . # ############################################################################### import requests from itertools import chain from odoo import api, fields, models USER_PRIVATE_FIELDS = ['password'] concat = chain.from_iterable class ResUsers(models.Model): """Inherits ResUsers to create records inside LoginLog""" _inherit = 'res.users' @api.model def _check_credentials(self, password, user_agent_env): """Supering the function and _check_credentials to create record inside LoginLog""" result = super()._check_credentials(password, user_agent_env) have_api_key = self.env['ir.config_parameter'].sudo().get_param( 'export_delete_login_log.have_api_key') api_key = self.env['ir.config_parameter'].sudo().get_param( 'export_delete_login_log.ipapi_key') ip_address = (requests.get( f'https://api.ipify.org?format=json').json()).get('ip') response = requests.get( f"https://ipapi.co/{ip_address}/json/?access_key={api_key}").json()\ if have_api_key else requests.get( f"https://ipapi.co/{ip_address}/json/").json() if response.get("error"): ip_data = { "ip": ip_address, } else: ip_data = { "ip": ip_address, "latitude": response.get("latitude"), "longitude": response.get("longitude"), "city": response.get("city"), "region": response.get("region"), "country": response.get("country_name"), "postal": response.get("postal"), "timezone": response.get("timezone"), "error": response.get("error"), "reason": response.get("reason"), } self.env['login.log'].sudo().create( {'name': self.name, 'ip_address': ip_address, 'geo_loc': str(ip_data["latitude"]) + ", " + str( ip_data["longitude"]), 'address': str(ip_data["city"]) + ", " + str( ip_data["region"]) + ", " + str(ip_data["country"]), 'postal_code': ip_data["postal"], 'time_zone': ip_data["timezone"], 'remark': ("Free quota exceeded" if ip_data["reason"] == "RateLimited" else ip_data["reason"]) if ip_data["error"] else None }) return result class LoginLog(models.Model): """Model to log information about user login""" _name = 'login.log' _description = 'Login Log' name = fields.Char(string="User Name", readonly=True) date_time = fields.Datetime(string="Login Date And Time", default=lambda self: fields.datetime.now(), readonly=True) ip_address = fields.Char(string="IP Address", readonly=True) geo_loc = fields.Char(string="Latitude / Longitude", readonly=True, help="Data is fetched using users IP address so the " "it may not be 100% precise") address = fields.Char(string="Address", readonly=True, help="Data is fetched using users IP address so the " "it may not be 100% precise") postal_code = fields.Char(string="Postal Code", readonly=True, help="Data is fetched using users IP address so " "the it may not be 100% precise") time_zone = fields.Char(string="Time Zone", readonly=True, help="Data is fetched using users IP address so " "the it may not be 100% precise") remark = fields.Text(string="Remarks", help="Remark is added when there is an error in the " "IP Address or other values, Error Reason: " "\n1) RateLimited: Free daily quota exceeded" "\n2) Invalid IP Address: The IP address used " "by the User is Invalid" "\n3) Reserved IP Address: The IP address used " "by the User is Reserved")