res_company.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. # -*- coding: utf-8 -*-
  2. # Part of Odoo. See LICENSE file for full copyright and licensing details.
  3. from odoo import fields, models, api, _
  4. from odoo.exceptions import UserError
  5. class ResCompany(models.Model):
  6. _inherit = 'res.company'
  7. google_drive_crm_folder_id = fields.Char(
  8. string='Google Drive CRM Folder ID',
  9. help='ID del folder en Google Drive para documentos del CRM'
  10. )
  11. google_drive_crm_folder_name = fields.Char(
  12. string='Google Drive CRM Folder Name',
  13. help='Nombre del folder en Google Drive para documentos del CRM',
  14. readonly=True
  15. )
  16. google_drive_crm_enabled = fields.Boolean(
  17. string='Enable Google Drive CRM Integration',
  18. default=False,
  19. help='Habilitar integración con Google Drive para CRM'
  20. )
  21. google_drive_crm_stage_id = fields.Many2one(
  22. 'crm.stage',
  23. string='CRM Stage for Google Drive Folder Creation',
  24. help='Etapa del CRM en la que se creará automáticamente la carpeta en Google Drive'
  25. )
  26. google_drive_crm_field_id = fields.Many2one(
  27. 'ir.model.fields',
  28. string='Google Drive Field',
  29. domain=[('model', '=', 'crm.lead')],
  30. help='Campo opcional de crm.lead que contiene información de Google Drive'
  31. )
  32. @api.onchange('google_drive_crm_folder_id')
  33. def _onchange_google_drive_crm_folder_id(self):
  34. """Update folder name when folder ID changes"""
  35. if self.google_drive_crm_folder_id:
  36. # TODO: Implement Google Drive API call to get folder name
  37. # For now, just clear the name
  38. self.google_drive_crm_folder_name = False
  39. def action_test_google_drive_connection(self):
  40. """Test Google Drive connection for this company"""
  41. self.ensure_one()
  42. if not self.google_drive_crm_enabled:
  43. raise UserError(_('Google Drive CRM Integration is not enabled for this company'))
  44. if not self.google_drive_crm_folder_id:
  45. raise UserError(_('Please set a Google Drive CRM Folder ID first'))
  46. try:
  47. import requests
  48. import json
  49. # Get Google API credentials from system parameters
  50. google_api_enabled = self.env['ir.config_parameter'].sudo().get_param('google_api.enabled', 'False')
  51. google_api_client_id = self.env['ir.config_parameter'].sudo().get_param('google_api.client_id', '')
  52. google_api_client_secret = self.env['ir.config_parameter'].sudo().get_param('google_api.client_secret', '')
  53. if not google_api_enabled or google_api_enabled == 'False':
  54. raise UserError(_('Google API Integration is not enabled in system settings'))
  55. if not google_api_client_id or not google_api_client_secret:
  56. raise UserError(_('Google API credentials are not configured in system settings'))
  57. # Get manual OAuth token
  58. access_token = self.env['ir.config_parameter'].sudo().get_param('google_api.access_token')
  59. if not access_token:
  60. raise UserError(_('No OAuth token found. Please connect your Google account in Google API settings first.'))
  61. # Validate folder ID format (Google Drive folder IDs are typically 33 characters)
  62. if len(self.google_drive_crm_folder_id) < 10 or len(self.google_drive_crm_folder_id) > 50:
  63. raise UserError(_('Google Drive Folder ID format appears to be invalid (should be 10-50 characters)'))
  64. # Test Google Drive API access with OAuth token
  65. headers = {
  66. 'Authorization': f'Bearer {access_token}',
  67. }
  68. # Test access to the specific folder
  69. drive_api_url = f"https://www.googleapis.com/drive/v3/files/{self.google_drive_crm_folder_id}"
  70. drive_response = requests.get(drive_api_url, headers=headers, timeout=10)
  71. if drive_response.status_code == 200:
  72. folder_data = drive_response.json()
  73. folder_name = folder_data.get('name', 'Unknown')
  74. return {
  75. 'type': 'ir.actions.client',
  76. 'tag': 'display_notification',
  77. 'params': {
  78. 'title': _('Success'),
  79. 'message': _('Google Drive connection test successful! Folder "%s" is accessible.') % folder_name,
  80. 'type': 'success',
  81. 'sticky': False,
  82. }
  83. }
  84. elif drive_response.status_code == 404:
  85. raise UserError(_('Google Drive folder not found. Please verify the Folder ID is correct.'))
  86. elif drive_response.status_code == 403:
  87. raise UserError(_('Access denied to Google Drive folder. Please check folder permissions.'))
  88. elif drive_response.status_code == 401:
  89. raise UserError(_('OAuth token expired or invalid. Please reconnect your Google account in Google API settings.'))
  90. else:
  91. raise UserError(_('Google Drive API test failed. Status: %s') % drive_response.status_code)
  92. except requests.exceptions.Timeout:
  93. raise UserError(_('Connection timeout. Please check your internet connection.'))
  94. except requests.exceptions.ConnectionError:
  95. raise UserError(_('Connection error. Please check your internet connection.'))
  96. except Exception as e:
  97. raise UserError(_('Google Drive connection test failed: %s') % str(e))
  98. def action_open_google_drive_folder(self):
  99. """Open Google Drive folder in browser"""
  100. self.ensure_one()
  101. if not self.google_drive_crm_folder_id:
  102. raise UserError(_('No Google Drive CRM folder configured'))
  103. folder_url = f"https://drive.google.com/drive/folders/{self.google_drive_crm_folder_id}"
  104. return {
  105. 'type': 'ir.actions.act_url',
  106. 'url': folder_url,
  107. 'target': 'new',
  108. }
  109. def action_create_google_drive_folder(self):
  110. """Create a folder in Google Drive using manual OAuth flow"""
  111. self.ensure_one()
  112. if not self.google_drive_crm_enabled:
  113. raise UserError(_('Google Drive CRM Integration is not enabled for this company'))
  114. # Get manual OAuth token
  115. access_token = self.env['ir.config_parameter'].sudo().get_param('google_api.access_token')
  116. if not access_token:
  117. raise UserError(_('No OAuth token found. Please connect your Google account in Google API settings first.'))
  118. try:
  119. import requests
  120. from datetime import datetime
  121. # Test Google Drive API access with OAuth token
  122. headers = {
  123. 'Authorization': f'Bearer {access_token}',
  124. 'Content-Type': 'application/json'
  125. }
  126. # Create folder metadata
  127. folder_name = f"CRM Folder - {self.name} - {datetime.now().strftime('%Y-%m-%d %H:%M')}"
  128. folder_metadata = {
  129. 'name': folder_name,
  130. 'mimeType': 'application/vnd.google-apps.folder'
  131. }
  132. # Add parent folder if specified
  133. if self.google_drive_crm_folder_id:
  134. folder_metadata['parents'] = [self.google_drive_crm_folder_id]
  135. # Create folder
  136. response = requests.post(
  137. 'https://www.googleapis.com/drive/v3/files',
  138. headers=headers,
  139. json=folder_metadata,
  140. timeout=30
  141. )
  142. if response.status_code == 200:
  143. folder_data = response.json()
  144. return {
  145. 'type': 'ir.actions.client',
  146. 'tag': 'display_notification',
  147. 'params': {
  148. 'title': _('Success'),
  149. 'message': _('Google Drive folder created successfully: %s') % folder_data.get('name'),
  150. 'type': 'success',
  151. 'sticky': False,
  152. }
  153. }
  154. elif response.status_code == 401:
  155. raise UserError(_('OAuth token expired or invalid. Please reconnect your Google account in Google API settings.'))
  156. else:
  157. raise UserError(_('Failed to create Google Drive folder. Status: %s') % response.status_code)
  158. except Exception as e:
  159. raise UserError(_('Failed to create Google Drive folder: %s') % str(e))
  160. def action_list_google_drive_folders(self):
  161. """List folders in Google Drive using manual OAuth flow"""
  162. self.ensure_one()
  163. if not self.google_drive_crm_enabled:
  164. raise UserError(_('Google Drive CRM Integration is not enabled for this company'))
  165. # Get manual OAuth token
  166. access_token = self.env['ir.config_parameter'].sudo().get_param('google_api.access_token')
  167. if not access_token:
  168. raise UserError(_('No OAuth token found. Please connect your Google account in Google API settings first.'))
  169. try:
  170. import requests
  171. headers = {
  172. 'Authorization': f'Bearer {access_token}',
  173. }
  174. # List folders
  175. params = {
  176. 'q': "mimeType='application/vnd.google-apps.folder' and trashed=false",
  177. 'fields': 'files(id,name,webViewLink)',
  178. 'pageSize': 10
  179. }
  180. response = requests.get(
  181. 'https://www.googleapis.com/drive/v3/files',
  182. headers=headers,
  183. params=params,
  184. timeout=30
  185. )
  186. if response.status_code == 200:
  187. data = response.json()
  188. folders = data.get('files', [])
  189. return {
  190. 'type': 'ir.actions.client',
  191. 'tag': 'display_notification',
  192. 'params': {
  193. 'title': _('Success'),
  194. 'message': _('Found %d folders in Google Drive.') % len(folders),
  195. 'type': 'success',
  196. 'sticky': False,
  197. }
  198. }
  199. elif response.status_code == 401:
  200. raise UserError(_('OAuth token expired or invalid. Please reconnect your Google account in Google API settings.'))
  201. else:
  202. raise UserError(_('Failed to list Google Drive folders. Status: %s') % response.status_code)
  203. except Exception as e:
  204. raise UserError(_('Failed to list Google Drive folders: %s') % str(e))