easyappointments/application/libraries/Synchronization.php

270 lines
9.6 KiB
PHP
Raw Normal View History

<?php defined('BASEPATH') or exit('No direct script access allowed');
/* ----------------------------------------------------------------------------
2022-01-18 15:05:42 +03:00
* Easy!Appointments - Online Appointment Scheduler
*
* @package EasyAppointments
* @author A.Tselegidis <alextselegidis@gmail.com>
2021-12-18 19:43:45 +03:00
* @copyright Copyright (c) Alex Tselegidis
* @license https://opensource.org/licenses/GPL-3.0 - GPLv3
* @link https://easyappointments.org
* @since v1.4.0
* ---------------------------------------------------------------------------- */
/**
2021-11-06 18:21:27 +03:00
* Synchronization library.
*
* Handles external calendar synchronization functionality.
*
* @package Libraries
*/
class Synchronization
{
/**
2023-03-13 11:06:18 +03:00
* @var EA_Controller|CI_Controller
*/
2023-03-13 11:06:18 +03:00
protected EA_Controller|CI_Controller $CI;
/**
* Synchronization constructor.
*/
public function __construct()
{
$this->CI = &get_instance();
$this->CI->load->model('providers_model');
$this->CI->load->model('appointments_model');
$this->CI->load->library('google_sync');
}
/**
* Synchronize changes made to the appointment with external calendars.
*
* @param array $appointment Appointment record.
* @param array $service Service record.
* @param array $provider Provider record.
* @param array $customer Customer record.
* @param array $settings Required settings for the notification content.
*/
public function sync_appointment_saved(
array $appointment,
array $service,
array $provider,
array $customer,
array $settings,
): void {
try {
// Google
if ($provider['settings']['google_sync']) {
if (empty($provider['settings']['google_token'])) {
throw new RuntimeException('No google token available for the provider: ' . $provider['id']);
}
$google_token = json_decode($provider['settings']['google_token'], true);
$this->CI->google_sync->refresh_token($google_token['refresh_token']);
if (empty($appointment['id_google_calendar'])) {
$google_event = $this->CI->google_sync->add_appointment(
$appointment,
$provider,
$service,
$customer,
$settings,
);
$appointment['id_google_calendar'] = $google_event->getId();
$this->CI->appointments_model->save($appointment);
} else {
$this->CI->google_sync->update_appointment($appointment, $provider, $service, $customer, $settings);
}
}
// CalDAV
if ($provider['settings']['caldav_sync']) {
$appointment['id_caldav_calendar'] = $this->CI->caldav_sync->save_appointment(
$appointment,
$service,
$provider,
$customer,
);
$this->CI->appointments_model->save($appointment);
}
} catch (Throwable $e) {
log_message(
'error',
'Synchronization - Could not sync confirmation details of appointment (' .
($appointment['id'] ?? '-') .
') : ' .
$e->getMessage(),
);
log_message('error', $e->getTraceAsString());
}
}
/**
* Synchronize removal of an appointment with external calendars.
*
* @param array $appointment Appointment record.
* @param array $provider Provider record.
*/
public function sync_appointment_deleted(array $appointment, array $provider): void
{
try {
// Google
if ($provider['settings']['google_sync'] && !empty($appointment['id_google_calendar'])) {
if (empty($provider['settings']['google_token'])) {
throw new RuntimeException('No google token available for the provider: ' . $provider['id']);
}
$google_token = json_decode($provider['settings']['google_token'], true);
$this->CI->google_sync->refresh_token($google_token['refresh_token']);
$this->CI->google_sync->delete_appointment($provider, $appointment['id_google_calendar']);
}
// CalDAV
if ($provider['settings']['caldav_sync'] && !empty($appointment['id_caldav_calendar'])) {
$this->CI->caldav_sync->delete_event($provider, $appointment['id_caldav_calendar']);
}
} catch (Throwable $e) {
log_message(
'error',
'Synchronization - Could not sync cancellation details of appointment (' .
($appointment['id'] ?? '-') .
') : ' .
$e->getMessage(),
);
log_message('error', $e->getTraceAsString());
}
}
/**
* Synchronize changes made to the unavailability with external calendars.
*
* @param array $unavailability Unavailability record.
* @param array $provider Provider record.
*/
2023-03-13 11:06:18 +03:00
public function sync_unavailability_saved(array $unavailability, array $provider): void
{
try {
// Google
if ($provider['settings']['google_sync']) {
if (empty($provider['settings']['google_token'])) {
throw new RuntimeException('No google token available for the provider: ' . $provider['id']);
}
$google_token = json_decode($provider['settings']['google_token'], true);
$this->CI->google_sync->refresh_token($google_token['refresh_token']);
if (empty($unavailability['id_google_calendar'])) {
$google_event = $this->CI->google_sync->add_unavailability($provider, $unavailability);
$unavailability['id_google_calendar'] = $google_event->getId();
$this->CI->unavailabilities_model->save($unavailability);
} else {
$this->CI->google_sync->update_unavailability($provider, $unavailability);
}
}
// CalDAV
if ($provider['settings']['caldav_sync']) {
$unavailability['id_caldav_calendar'] = $this->CI->caldav_sync->save_unavailability(
$unavailability,
$provider,
);
$this->CI->unavailabilities_model->save($unavailability);
}
} catch (Throwable $e) {
log_message(
'error',
'Synchronization - Could not sync cancellation details of unavailability (' .
($appointment['id'] ?? '-') .
') : ' .
$e->getMessage(),
);
log_message('error', $e->getTraceAsString());
}
}
/**
* Synchronize removal of an unavailability with external calendars.
*
* @param array $unavailability Unavailability record.
* @param array $provider Provider record.
*/
2023-03-13 11:06:18 +03:00
public function sync_unavailability_deleted(array $unavailability, array $provider): void
{
try {
// Google
if ($provider['settings']['google_sync'] && !empty($unavailability['id_google_calendar'])) {
if (empty($provider['settings']['google_token'])) {
throw new RuntimeException('No google token available for the provider: ' . $provider['id']);
}
$google_token = json_decode($provider['settings']['google_token'], true);
$this->CI->google_sync->refresh_token($google_token['refresh_token']);
$this->CI->google_sync->delete_unavailability($provider, $unavailability['id_google_calendar']);
}
// CalDAV
if ($provider['settings']['caldav_sync'] && !empty($unavailability['id_caldav_calendar'])) {
$this->CI->caldav_sync->delete_event($provider, $unavailability['id_caldav_calendar']);
}
} catch (Throwable $e) {
log_message(
'error',
'Synchronization - Could not sync cancellation details of unavailability (' .
($appointment['id'] ?? '-') .
') : ' .
$e->getMessage(),
);
log_message('error', $e->getTraceAsString());
}
}
/**
* Make sure a synced appointment is removed from Google/CalDAV Calendar, if its provider is changed.
*
* @throws Exception
*/
2023-03-13 11:06:18 +03:00
public function remove_appointment_on_provider_change($appointment_id): void
{
2023-03-13 11:06:18 +03:00
$existing_appointment = $this->CI->appointments_model->find($appointment_id);
$existing_google_id = $existing_appointment['id_google_calendar'];
$existing_caldav_id = $existing_appointment['id_caldav_calendar'];
$existing_provider_id = $existing_appointment['id_users_provider'];
if (
(!empty($existing_google_id) || !empty($existing_caldav_id)) &&
(int) $existing_provider_id !== (int) $existing_appointment['id_users_provider']
) {
2023-03-13 11:06:18 +03:00
$existing_provider = $this->CI->providers_model->find($existing_provider_id);
if ($existing_provider['settings']['google_sync'] || $existing_provider['settings']['caldav_sync']) {
$this->sync_appointment_deleted($existing_appointment, $existing_provider);
}
}
}
}