mirror of
https://github.com/alextselegidis/easyappointments.git
synced 2024-12-11 17:22:23 +03:00
412 lines
15 KiB
PHP
412 lines
15 KiB
PHP
<?php defined('BASEPATH') or exit('No direct script access allowed');
|
|
|
|
/* ----------------------------------------------------------------------------
|
|
* Easy!Appointments - Open Source Web Scheduler
|
|
*
|
|
* @package EasyAppointments
|
|
* @author A.Tselegidis <alextselegidis@gmail.com>
|
|
* @copyright Copyright (c) 2013 - 2020, Alex Tselegidis
|
|
* @license http://opensource.org/licenses/GPL-3.0 - GPLv3
|
|
* @link http://easyappointments.org
|
|
* @since v1.0.0
|
|
* ---------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
* Class Google_sync
|
|
*
|
|
* This class implements all the core synchronization between the Google Calendar and the Easy!Appointments system.
|
|
*
|
|
* Do not place any model handling inside this library.
|
|
*
|
|
* @package Libraries
|
|
*/
|
|
class Google_sync {
|
|
/**
|
|
* CodeIgniter Instance
|
|
*
|
|
* @var EA_Controller
|
|
*/
|
|
protected $CI;
|
|
|
|
/**
|
|
* Google API Client
|
|
*
|
|
* @var Google_Client
|
|
*/
|
|
protected $client;
|
|
|
|
/**
|
|
* Google Calendar Service
|
|
*
|
|
* @var Google_Service_Calendar
|
|
*/
|
|
protected $service;
|
|
|
|
/**
|
|
* Class Constructor
|
|
*
|
|
* This method initializes the Google client class and the Calendar service
|
|
* class so that they can be used by the other methods.
|
|
*/
|
|
public function __construct()
|
|
{
|
|
$this->CI =& get_instance();
|
|
|
|
// Initialize google client and calendar service.
|
|
$http = new GuzzleHttp\Client([
|
|
'verify' => false
|
|
]);
|
|
|
|
$this->client = new Google_Client();
|
|
$this->client->setHttpClient($http);
|
|
$this->client->setApplicationName(config('google_application_name'));
|
|
$this->client->setClientId(config('google_client_id'));
|
|
$this->client->setClientSecret(config('google_client_secret'));
|
|
$this->client->setDeveloperKey(config('google_api_key'));
|
|
$this->client->setRedirectUri(site_url('google/oauth_callback'));
|
|
$this->client->setPrompt('consent');
|
|
$this->client->setAccessType('offline');
|
|
$this->client->addScope([
|
|
Google_Service_Calendar::CALENDAR,
|
|
]);
|
|
|
|
$this->service = new Google_Service_Calendar($this->client);
|
|
}
|
|
|
|
/**
|
|
* Get Google OAuth authorization url.
|
|
*
|
|
* This url must be used to redirect the user to the Google user consent page,
|
|
* where the user grants access to his data for the Easy!Appointments app.
|
|
*/
|
|
public function get_auth_url()
|
|
{
|
|
// "max_auth_age" is needed because the user needs to always log in
|
|
// and not use an existing session.
|
|
return $this->client->createAuthUrl() . '&max_auth_age=0';
|
|
}
|
|
|
|
/**
|
|
* Authenticate the Google API usage.
|
|
*
|
|
* When the user grants consent for his data usage, google is going to redirect
|
|
* the browser back to the given redirect url. There a authentication code is
|
|
* provided. Using this code, we can authenticate the API usage and store the
|
|
* token information to the database.
|
|
*
|
|
* @param $code
|
|
*
|
|
* @return array
|
|
*
|
|
* @throws Exception
|
|
*/
|
|
public function authenticate($code)
|
|
{
|
|
$response = $this->client->fetchAccessTokenWithAuthCode($code);
|
|
|
|
if (isset($response['error']))
|
|
{
|
|
throw new Exception('Google Authentication Error (' . $response['error'] . '): ' . $response['error_description']);
|
|
}
|
|
|
|
return $response;
|
|
}
|
|
|
|
/**
|
|
* Refresh the Google Client access token.
|
|
*
|
|
* This method must be executed every time we need to make actions on a
|
|
* provider's Google Calendar account. A new token is necessary and the
|
|
* only way to get it is to use the stored refresh token that was provided
|
|
* when the provider granted consent to Easy!Appointments for use his
|
|
* Google Calendar account.
|
|
*
|
|
* @param string $refresh_token The provider's refresh token. This value is
|
|
* stored in the database and used every time we need to make actions to his
|
|
* Google Caledar account.
|
|
*/
|
|
public function refresh_token($refresh_token)
|
|
{
|
|
$this->client->refreshToken($refresh_token);
|
|
}
|
|
|
|
/**
|
|
* Add an appointment record to its providers Google Calendar account.
|
|
*
|
|
* This method checks whether the appointment's provider has enabled the Google Sync utility of Easy!Appointments
|
|
* and the stored access token is still valid. If yes, the selected appointment record is going to be added to the
|
|
* Google Calendar account.
|
|
*
|
|
* @param array $appointment Contains the appointment record data.
|
|
* @param array $provider Contains the provider record data.
|
|
* @param array $service Contains the service record data.
|
|
* @param array $customer Contains the customer recod data.
|
|
* @param array $settings Contains some company settings that are used by this method.
|
|
*
|
|
* @return Google_Service_Calendar_Event Returns the Google_Event class object.
|
|
*/
|
|
public function add_appointment($appointment, $provider, $service, $customer, $settings)
|
|
{
|
|
$this->CI->load->helper('general');
|
|
|
|
$event = new Google_Service_Calendar_Event();
|
|
$event->setSummary(($service != NULL) ? $service['name'] : 'Unavailable');
|
|
$event->setDescription($appointment['notes']);
|
|
$event->setLocation(isset($appointment['location']) ? $appointment['location'] : $settings['company_name']);
|
|
|
|
$timezone = new DateTimeZone($provider['timezone']);
|
|
|
|
$start = new Google_Service_Calendar_EventDateTime();
|
|
$start->setDateTime((new DateTime($appointment['start_datetime'], $timezone))->format(DateTime::RFC3339));
|
|
$event->setStart($start);
|
|
|
|
$end = new Google_Service_Calendar_EventDateTime();
|
|
$end->setDateTime((new DateTime($appointment['end_datetime'], $timezone))->format(DateTime::RFC3339));
|
|
$event->setEnd($end);
|
|
|
|
$event->attendees = [];
|
|
|
|
$event_provider = new Google_Service_Calendar_EventAttendee();
|
|
$event_provider->setDisplayName($provider['first_name'] . ' '
|
|
. $provider['last_name']);
|
|
$event_provider->setEmail($provider['email']);
|
|
$event->attendees[] = $event_provider;
|
|
|
|
if ($customer != NULL)
|
|
{
|
|
$event_customer = new Google_Service_Calendar_EventAttendee();
|
|
$event_customer->setDisplayName($customer['first_name'] . ' ' . $customer['last_name']);
|
|
$event_customer->setEmail($customer['email']);
|
|
$event->attendees[] = $event_customer;
|
|
}
|
|
|
|
// Add the new event to the google calendar.
|
|
$created_event = $this->service->events->insert($provider['settings']['google_calendar'], $event);
|
|
|
|
return $created_event;
|
|
}
|
|
|
|
/**
|
|
* Update an existing appointment that is already synced with Google Calendar.
|
|
*
|
|
* This method updates the google calendar event item that is connected with the
|
|
* provided appointment record of Easy!Appointments.
|
|
*
|
|
* @param array $appointment Contains the appointment record data.
|
|
* @param array $provider Contains the provider record data.
|
|
* @param array $service Contains the service record data.
|
|
* @param array $customer Contains the customer recod data.
|
|
* @parma array $settings Contains some company settings that are used by this method.
|
|
*
|
|
* @return Google_Service_Calendar_Event Returns the Google_Service_Calendar_Event class object.
|
|
*/
|
|
public function update_appointment($appointment, $provider, $service, $customer, $settings)
|
|
{
|
|
$this->CI->load->helper('general');
|
|
|
|
$event = $this->service->events->get($provider['settings']['google_calendar'],
|
|
$appointment['id_google_calendar']);
|
|
|
|
$event->setSummary($service['name']);
|
|
$event->setDescription($appointment['notes']);
|
|
$event->setLocation(isset($appointment['location']) ? $appointment['location'] : $settings['company_name']);
|
|
|
|
$timezone = new DateTimeZone($provider['timezone']);
|
|
|
|
$start = new Google_Service_Calendar_EventDateTime();
|
|
$start->setDateTime((new DateTime($appointment['start_datetime'], $timezone))->format(DateTime::RFC3339));
|
|
$event->setStart($start);
|
|
|
|
$end = new Google_Service_Calendar_EventDateTime();
|
|
$end->setDateTime((new DateTime($appointment['end_datetime'], $timezone))->format(DateTime::RFC3339));
|
|
$event->setEnd($end);
|
|
|
|
$event->attendees = [];
|
|
|
|
$event_provider = new Google_Service_Calendar_EventAttendee();
|
|
$event_provider->setDisplayName($provider['first_name'] . ' ' . $provider['last_name']);
|
|
$event_provider->setEmail($provider['email']);
|
|
$event->attendees[] = $event_provider;
|
|
|
|
if ($customer != NULL)
|
|
{
|
|
$event_customer = new Google_Service_Calendar_EventAttendee();
|
|
$event_customer->setDisplayName($customer['first_name'] . ' '
|
|
. $customer['last_name']);
|
|
$event_customer->setEmail($customer['email']);
|
|
$event->attendees[] = $event_customer;
|
|
}
|
|
|
|
$updated_event = $this->service->events->update($provider['settings']['google_calendar'],
|
|
$event->getId(), $event);
|
|
|
|
return $updated_event;
|
|
}
|
|
|
|
/**
|
|
* Delete an existing appointment from Google Calendar.
|
|
*
|
|
* @param array $provider Contains the provider record data.
|
|
* @param string $google_event_id The Google Calendar event id to
|
|
* be deleted.
|
|
*/
|
|
public function delete_appointment($provider, $google_event_id)
|
|
{
|
|
try
|
|
{
|
|
$this->service->events->delete($provider['settings']['google_calendar'], $google_event_id);
|
|
}
|
|
catch (Exception $ex)
|
|
{
|
|
// Event was not found on Google Calendar.
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Add unavailable period event to Google Calendar.
|
|
*
|
|
* @param array $provider Contains the provider record data.
|
|
* @param array $unavailable Contains unavailable period's data.
|
|
*
|
|
* @return Google_Service_Calendar_Event Returns the google event's object.
|
|
*/
|
|
public function add_unavailable($provider, $unavailable)
|
|
{
|
|
$this->CI->load->helper('general');
|
|
|
|
$event = new Google_Service_Calendar_Event();
|
|
$event->setSummary('Unavailable');
|
|
$event->setDescription($unavailable['notes']);
|
|
|
|
$timezone = new DateTimeZone($provider['timezone']);
|
|
|
|
$start = new Google_Service_Calendar_EventDateTime();
|
|
$start->setDateTime((new DateTime($unavailable['start_datetime'], $timezone))->format(DateTime::RFC3339));
|
|
$event->setStart($start);
|
|
|
|
$end = new Google_Service_Calendar_EventDateTime();
|
|
$end->setDateTime((new DateTime($unavailable['end_datetime'], $timezone))->format(DateTime::RFC3339));
|
|
$event->setEnd($end);
|
|
|
|
// Add the new event to the google calendar.
|
|
$created_event = $this->service->events->insert($provider['settings']['google_calendar'], $event);
|
|
|
|
return $created_event;
|
|
|
|
}
|
|
|
|
/**
|
|
* Update Google Calendar unavailable period event.
|
|
*
|
|
* @param array $provider Contains the provider record data.
|
|
* @param array $unavailable Contains the unavailable period data.
|
|
*
|
|
* @return Google_Service_Calendar_Event Returns the Google_Service_Calendar_Event object.
|
|
*/
|
|
public function update_unavailable($provider, $unavailable)
|
|
{
|
|
$this->CI->load->helper('general');
|
|
|
|
$event = $this->service->events->get($provider['settings']['google_calendar'],
|
|
$unavailable['id_google_calendar']);
|
|
$event->setDescription($unavailable['notes']);
|
|
|
|
$timezone = new DateTimeZone($provider['timezone']);
|
|
|
|
$start = new Google_Service_Calendar_EventDateTime();
|
|
$start->setDateTime((new DateTime($unavailable['start_datetime'], $timezone))->format(DateTime::RFC3339));
|
|
$event->setStart($start);
|
|
|
|
$end = new Google_Service_Calendar_EventDateTime();
|
|
$end->setDateTime((new DateTime($unavailable['end_datetime'], $timezone))->format(DateTime::RFC3339));
|
|
$event->setEnd($end);
|
|
|
|
$updated_event = $this->service->events->update($provider['settings']['google_calendar'],
|
|
$event->getId(), $event);
|
|
|
|
return $updated_event;
|
|
}
|
|
|
|
/**
|
|
* Delete unavailable period event from Google Calendar.
|
|
*
|
|
* @param array $provider Contains the provider record data.
|
|
* @param string $google_event_id Google Calendar event id to be deleted.
|
|
*/
|
|
public function delete_unavailable($provider, $google_event_id)
|
|
{
|
|
try
|
|
{
|
|
$this->service->events->delete($provider['settings']['google_calendar'], $google_event_id);
|
|
}
|
|
catch (Exception $ex)
|
|
{
|
|
// Event was not found on Google Calendar.
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get an event object from gcal
|
|
*
|
|
* @param array $provider Contains the provider record data.
|
|
* @param string $google_event_id Id of the google calendar event.
|
|
*
|
|
* @return Google_Service_Calendar_Event Returns the google event object.
|
|
*/
|
|
public function get_event($provider, $google_event_id)
|
|
{
|
|
return $this->service->events->get($provider['settings']['google_calendar'], $google_event_id);
|
|
}
|
|
|
|
/**
|
|
* Get all the events between the sync period.
|
|
*
|
|
* @param string $google_calendar The name of the google calendar to be used.
|
|
* @param string $start The start date of sync period.
|
|
* @param string $end The end date of sync period.
|
|
*
|
|
* @return object Returns an array with Google_Service_Calendar_Event objects that belong on the given
|
|
* sync period (start, end).
|
|
*/
|
|
public function get_sync_events($google_calendar, $start, $end)
|
|
{
|
|
$this->CI->load->helper('general');
|
|
|
|
$params = [
|
|
'timeMin' => date(DateTime::RFC3339, $start),
|
|
'timeMax' => date(DateTime::RFC3339, $end),
|
|
'singleEvents' => TRUE,
|
|
];
|
|
|
|
return $this->service->events->listEvents($google_calendar, $params);
|
|
}
|
|
|
|
/**
|
|
* Return available google calendars for specific user.
|
|
*
|
|
* The given user's token must already exist in db in order to get access to his
|
|
* Google Calendar account.
|
|
*
|
|
* @param string $google_token The user's token will be used to grant access to google calendar.
|
|
*
|
|
* @return array Returns an array with the available calendars.
|
|
*/
|
|
public function get_google_calendars()
|
|
{
|
|
$calendarList = $this->service->calendarList->listCalendarList();
|
|
$calendars = [];
|
|
foreach ($calendarList->items as $google_calendar)
|
|
{
|
|
if ($google_calendar->getAccessRole() === 'reader')
|
|
{
|
|
continue;
|
|
}
|
|
|
|
$calendars[] = [
|
|
'id' => $google_calendar->id,
|
|
'summary' => $google_calendar->summary
|
|
];
|
|
}
|
|
return $calendars;
|
|
}
|
|
}
|