153 lines
4.8 KiB
Python
153 lines
4.8 KiB
Python
from werkzeug.wrappers import Response
|
|
import json
|
|
|
|
import frappe
|
|
from frappe import _
|
|
from .twilio_handler import Twilio, IncomingCall, TwilioCallDetails
|
|
|
|
@frappe.whitelist()
|
|
def generate_access_token():
|
|
"""Returns access token that is required to authenticate Twilio Client SDK.
|
|
"""
|
|
twilio = Twilio.connect()
|
|
if not twilio:
|
|
return {}
|
|
|
|
from_number = frappe.db.get_value('Twilio Agents', frappe.session.user, 'twilio_number')
|
|
if not from_number:
|
|
return {
|
|
"ok": False,
|
|
"error": "caller_phone_identity_missing",
|
|
"detail": "Phone number is not mapped to the caller"
|
|
}
|
|
|
|
token=twilio.generate_voice_access_token(identity=frappe.session.user)
|
|
return {
|
|
'token': frappe.safe_decode(token)
|
|
}
|
|
|
|
@frappe.whitelist(allow_guest=True)
|
|
def voice(**kwargs):
|
|
"""This is a webhook called by twilio to get instructions when the voice call request comes to twilio server.
|
|
"""
|
|
def _get_caller_number(caller):
|
|
identity = caller.replace('client:', '').strip()
|
|
user = Twilio.emailid_from_identity(identity)
|
|
return frappe.db.get_value('Twilio Agents', user, 'twilio_number')
|
|
|
|
args = frappe._dict(kwargs)
|
|
twilio = Twilio.connect()
|
|
if not twilio:
|
|
return
|
|
|
|
assert args.AccountSid == twilio.account_sid
|
|
assert args.ApplicationSid == twilio.application_sid
|
|
|
|
# Generate TwiML instructions to make a call
|
|
from_number = _get_caller_number(args.Caller)
|
|
resp = twilio.generate_twilio_dial_response(from_number, args.To)
|
|
|
|
call_details = TwilioCallDetails(args, call_from=from_number)
|
|
create_call_log(call_details)
|
|
return Response(resp.to_xml(), mimetype='text/xml')
|
|
|
|
@frappe.whitelist(allow_guest=True)
|
|
def twilio_incoming_call_handler(**kwargs):
|
|
args = frappe._dict(kwargs)
|
|
call_details = TwilioCallDetails(args)
|
|
create_call_log(call_details)
|
|
|
|
resp = IncomingCall(args.From, args.To).process()
|
|
return Response(resp.to_xml(), mimetype='text/xml')
|
|
|
|
def create_call_log(call_details: TwilioCallDetails):
|
|
call_log = frappe.get_doc({**call_details.to_dict(),
|
|
'doctype': 'CRM Call Log',
|
|
'medium': 'Twilio'
|
|
})
|
|
|
|
call_log.flags.ignore_permissions = True
|
|
call_log.save()
|
|
frappe.db.commit()
|
|
|
|
def update_call_log(call_sid, status=None):
|
|
"""Update call log status.
|
|
"""
|
|
twilio = Twilio.connect()
|
|
if not (twilio and frappe.db.exists("CRM Call Log", call_sid)): return
|
|
|
|
call_details = twilio.get_call_info(call_sid)
|
|
call_log = frappe.get_doc("CRM Call Log", call_sid)
|
|
call_log.status = status or TwilioCallDetails.get_call_status(call_details.status)
|
|
call_log.duration = call_details.duration
|
|
call_log.start_time = get_datetime_from_timestamp(call_details.start_time)
|
|
call_log.end_time = get_datetime_from_timestamp(call_details.end_time)
|
|
call_log.lead = get_lead_from_number(call_log)
|
|
if call_log.note and call_log.lead:
|
|
frappe.db.set_value("CRM Note", call_log.note, "lead", call_log.lead)
|
|
call_log.flags.ignore_permissions = True
|
|
call_log.save()
|
|
frappe.db.commit()
|
|
|
|
@frappe.whitelist(allow_guest=True)
|
|
def update_recording_info(**kwargs):
|
|
try:
|
|
args = frappe._dict(kwargs)
|
|
recording_url = args.RecordingUrl
|
|
call_sid = args.CallSid
|
|
update_call_log(call_sid)
|
|
frappe.db.set_value("CRM Call Log", call_sid, "recording_url", recording_url)
|
|
except:
|
|
frappe.log_error(title=_("Failed to capture Twilio recording"))
|
|
|
|
@frappe.whitelist(allow_guest=True)
|
|
def get_call_info(**kwargs):
|
|
"""This is a webhook called when the outgoing call status changes.
|
|
E.g. 'initiated' 'ringing', 'in-progress', 'completed' etc.
|
|
"""
|
|
args = frappe._dict(kwargs)
|
|
call_info = {
|
|
'ParentCallSid': args.ParentCallSid,
|
|
'CallSid': args.CallSid,
|
|
'CallStatus': args.CallStatus,
|
|
'CallDuration': args.CallDuration,
|
|
'From': args.From,
|
|
'To': args.To,
|
|
}
|
|
|
|
client = Twilio.get_twilio_client()
|
|
client.calls(args.ParentCallSid).user_defined_messages.create(
|
|
content=json.dumps(call_info)
|
|
)
|
|
|
|
def get_datetime_from_timestamp(timestamp):
|
|
from datetime import datetime
|
|
from pytz import timezone
|
|
|
|
datetime_utc_tz_str = timestamp.strftime('%Y-%m-%d %H:%M:%S%z')
|
|
datetime_utc_tz = datetime.strptime(datetime_utc_tz_str, '%Y-%m-%d %H:%M:%S%z')
|
|
system_timezone = frappe.utils.get_system_timezone()
|
|
converted_datetime = datetime_utc_tz.astimezone(timezone(system_timezone))
|
|
return frappe.utils.format_datetime(converted_datetime, 'yyyy-MM-dd HH:mm:ss')
|
|
|
|
@frappe.whitelist()
|
|
def add_note_to_call_log(call_sid, note):
|
|
"""Add note to call log. based on child call sid.
|
|
"""
|
|
twilio = Twilio.connect()
|
|
if not twilio: return
|
|
|
|
call_details = twilio.get_call_info(call_sid)
|
|
frappe.db.set_value("CRM Call Log", call_details.parent_call_sid, "note", note)
|
|
frappe.db.commit()
|
|
|
|
def get_lead_from_number(call):
|
|
"""Get lead from the given number.
|
|
"""
|
|
lead = None
|
|
if call.type == 'Outgoing':
|
|
lead = frappe.db.get_value("CRM Lead", { "mobile_no": call.get('to') })
|
|
else:
|
|
lead = frappe.db.get_value("CRM Lead", { "mobile_no": call.get('from') })
|
|
|
|
return lead |