You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							168 lines
						
					
					
						
							7.3 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							168 lines
						
					
					
						
							7.3 KiB
						
					
					
				| # -*- coding: utf-8 -*- | |
| ############################################################################# | |
| # | |
| #    Cybrosys Technologies Pvt. Ltd. | |
| # | |
| #    Copyright (C) 2023-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) | |
| #    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | |
| # | |
| #    You can modify it under the terms of the GNU LESSER | |
| #    GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. | |
| # | |
| #    You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE | |
| #    (LGPL v3) along with this program. | |
| #    If not, see <http://www.gnu.org/licenses/>. | |
| # | |
| ############################################################################## | |
| import logging | |
| import requests | |
| from odoo import fields, models, _ | |
| from odoo.exceptions import UserError, ValidationError | |
| 
 | |
| _logger = logging.getLogger(__name__) | |
| 
 | |
| TIMEOUT = 20 | |
| 
 | |
| 
 | |
| class ResCompany(models.Model): | |
|     """Inherit res_company for integrating contacts with Google""" | |
|     _inherit = "res.company" | |
| 
 | |
|     contact_client_id = fields.Char( | |
|         string="Client Id", help='People API Client ID') | |
|     contact_client_secret = fields.Char( | |
|         string="Client Secret", help='People API Client Secret') | |
|     contact_redirect_uri = fields.Char( | |
|         string="Authorized redirect URIs", | |
|         default="http://localhost:8016/google_contact_authentication", | |
|         help='People API Authorized redirect URIs') | |
|     contact_company_access_token = fields.Char( | |
|         string="Access Token", copy=False, | |
|         help='People API Access Token') | |
|     contact_company_access_token_expiry = fields.Datetime( | |
|         string="Token expiry", help='People API Token Expiry') | |
|     contact_company_refresh_token = fields.Char( | |
|         string="Refresh Token", copy=False, help='People API Refresh Token') | |
|     contact_company_authorization_code = fields.Char( | |
|         string="Authorization Code", help='People API Authorization Code') | |
| 
 | |
|     def action_google_contact_authenticate(self): | |
|         """Authenticate the connection to Google.""" | |
|         if not self.contact_client_id: | |
|             raise ValidationError("Please Enter Client ID") | |
|         if not self.contact_redirect_uri: | |
|             raise ValidationError("Please Enter Client Secret") | |
|         people_scope = 'https://www.googleapis.com/auth/contacts' | |
|         url = ( | |
|             "https://accounts.google.com/o/oauth2/v2/auth?response_type=code" | |
|             "&access_type=offline&client_id={}&redirect_uri={}&scope={} " | |
|         ).format(self.contact_client_id, | |
|                  self.contact_redirect_uri, people_scope) | |
|         return { | |
|             "type": 'ir.actions.act_url', | |
|             "url": url, | |
|             "target": "new" | |
|         } | |
| 
 | |
|     def action_google_contact_refresh_token(self): | |
|         """Request a refresh token from Google.""" | |
|         if not self.contact_client_id: | |
|             raise UserError( | |
|                 _('Client ID is not yet configured.')) | |
|         if not self.contact_client_secret: | |
|             raise UserError( | |
|                 _('Client Secret is not yet configured.')) | |
|         if not self.contact_company_refresh_token: | |
|             raise UserError( | |
|                 _('Refresh Token is not yet configured.')) | |
|         data = { | |
|             'client_id': self.contact_client_id, | |
|             'client_secret': self.contact_client_secret, | |
|             'refresh_token': self.contact_company_refresh_token, | |
|             'grant_type': 'refresh_token', | |
|         } | |
|         response = requests.post( | |
|             'https://accounts.google.com/o/oauth2/token', data=data, | |
|             headers={ | |
|                 'content-type': 'application/x-www-form-urlencoded'}, | |
|             timeout=TIMEOUT) | |
|         if response.json() and response.json().get('access_token'): | |
|             self.write({ | |
|                 'contact_company_access_token': | |
|                     response.json().get('access_token'), | |
|             }) | |
|         else: | |
|             raise UserError( | |
|                 _('Something went wrong during the token generation.' | |
|                   ' Please request again an authorization code.') | |
|             ) | |
| 
 | |
|     def action_import_google_contacts(self): | |
|         """IMPORT Contacts FROM Google TO ODOO""" | |
|         url = ("https://people.googleapis.com/v1/people/me/" | |
|                "connections?personFields=names,addresses," | |
|                "emailAddresses,phoneNumbers&pageSize=1000") | |
|         headers = { | |
|             'Authorization': f'Bearer {self.contact_company_access_token}', | |
|             'Content-Type': 'application/json' | |
|         } | |
|         response = requests.get(url, headers=headers) | |
|         if response.status_code == 200: | |
|             data = response.json().get('connections', []) | |
|             partners = [] | |
|             for connection in data: | |
|                 cnt_rsr_name = connection.get('resourceName', '') | |
|                 etag = connection.get('etag', '') | |
|                 names = connection.get('names', [{}])[0] | |
|                 first_name = names.get('givenName', '') | |
|                 last_name = names.get('familyName', '') | |
|                 name = names.get('displayName', '') | |
|                 emailAddresses = connection.get('emailAddresses', [{}])[0] | |
|                 email = emailAddresses.get('value', '') | |
|                 phoneNumbers = connection.get('phoneNumbers', [{}])[0] | |
|                 phone = phoneNumbers.get('value', '') | |
|                 addresses = connection.get('addresses', [{}])[0] | |
|                 street = addresses.get('streetAddress', '') | |
|                 street2 = addresses.get('extendedAddress', '') | |
|                 city = addresses.get('city', '') | |
|                 pin = addresses.get('postalCode', '') | |
|                 state = addresses.get('region', '') | |
|                 state = self.env['res.country.state'].search( | |
|                     [("name", 'ilike', state)], limit=1) | |
|                 state_id = state.id if state else False | |
|                 country_code = addresses.get('countryCode', '') | |
|                 country = self.env['res.country'].search( | |
|                     [('code', 'ilike', country_code)], limit=1) | |
|                 country_id = country.id if country else False | |
|                 partner_vals = { | |
|                     'name': name or '', | |
|                     'first_name': first_name or '', | |
|                     'last_name': last_name or '', | |
|                     'email': email or '', | |
|                     'street': street or '', | |
|                     'street2': street2 or '', | |
|                     'city': city or '', | |
|                     'zip': pin or '', | |
|                     'state_id': state_id or False, | |
|                     'country_id': country_id or False, | |
|                     'phone': phone, | |
|                     'google_resource': cnt_rsr_name, | |
|                     'google_etag': etag, | |
|                 } | |
|                 existing_partner = self.env['res.partner'].search( | |
|                     [('google_resource', '=', cnt_rsr_name)], limit=1) | |
|                 if existing_partner: | |
|                     existing_partner.write(partner_vals) | |
|                 else: | |
|                     partners.append(partner_vals) | |
|             if partners: | |
|                 self.env['res.partner'].create(partners) | |
|             _logger.info("Contact imported successfully!") | |
|         else: | |
|             error_message = f"Failed to import contact. Error: {response.text}" | |
|             raise ValidationError(error_message)
 | |
| 
 |