99 lines
4.0 KiB
Python

from odoo import http
from odoo.http import request
import json
import logging
_logger = logging.getLogger(__name__)
class UberWebhookController(http.Controller):
@http.route('/uber/webhook/delivery', type='json', auth='none', methods=['POST'], csrf=False)
def uber_delivery_webhook(self, **post):
"""Handle status updates from Uber Direct"""
data = json.loads(request.httprequest.data)
_logger.info("Uber Webhook Received: %s", json.dumps(data, indent=2))
uber_delivery_id = data.get('delivery_id')
status = data.get('status') # e.g., 'picked_up', 'delivered'
if uber_delivery_id:
order = request.env['pos.order'].sudo().search([('uber_delivery_id', '=', uber_delivery_id)], limit=1)
if order:
# Map Uber status to Odoo status
status_map = {
'pickup': 'pickup',
'pickup_completed': 'delivering',
'dropoff_completed': 'delivered',
'cancelled': 'cancelled'
}
order.uber_status = status_map.get(status, order.uber_status)
return {'status': 'success'}
return {'status': 'ignored'}
class UberDeliveryController(http.Controller):
@http.route('/shop/uber/quote', type='json', auth='public', website=True, csrf=False)
def uber_quote(self, address_data, **post):
"""Get Uber quote for a website address with cleaned address formatting"""
order = request.website.sale_get_order()
if not order:
return {'success': False, 'error': 'No active order'}
config = request.env['uber.config'].sudo().search([('active', '=', True)], limit=1)
if not config:
return {'success': False, 'error': 'Uber not configured'}
company = request.website.company_id
# Build STRUCTURED pickup address (Object) mapping POS exactly
pickup_address = {
"street_address": [company.street or ""],
"city": company.city or "",
"state": company.state_id.code if company.state_id else "",
"zip_code": company.zip or "",
"country": company.country_id.code or "CA"
}
# User entered address fields
street = (address_data.get('street') or '').strip()
street2 = (address_data.get('street2') or '').strip()
full_street = f"{street}, {street2}" if street2 else street
state_input = (address_data.get('state') or '').split('(')[0].strip()
state_record = request.env['res.country.state'].sudo().search([
('country_id.code', '=', 'CA'),
'|', ('name', '=ilike', state_input), ('code', '=ilike', state_input)
], limit=1)
state_code = state_record.code if state_record else "ON"
# Build STRUCTURED dropoff address (Object)
dropoff_address = {
"street_address": [full_street],
"city": address_data.get('city', '').strip(),
"state": state_code,
"zip_code": address_data.get('zip', '').strip(),
"country": "CA"
}
# For logging, still create strings
p_str = f"{pickup_address['street_address'][0]}, {pickup_address['city']} {pickup_address['state']}"
d_str = f"{dropoff_address['street_address'][0]}, {dropoff_address['city']} {dropoff_address['state']}"
_logger.info("WEBSITE UBER QUOTE (STRUCTURED) -\nPickup: [%s]\nDropoff: [%s]", p_str, d_str)
# POS ENCODING: The POS sends these as JSON-encoded STRINGS
result = config.get_uber_quote(json.dumps(pickup_address), json.dumps(dropoff_address))
if result.get('success'):
order.sudo()._add_uber_delivery_fee(result['fee_amount'])
return {
'success': True,
'fee': result['fee_amount'],
'eta': result.get('estimated_arrival'),
}
else:
_logger.warning("Uber Quote Failed: %s", result.get('error'))
return result