MaketRandevu/application/models/Appointments_model.php

610 lines
21 KiB
PHP
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<?php defined('BASEPATH') or exit('No direct script access allowed');
/* ----------------------------------------------------------------------------
* MaketRandevu - MAKET Randevu Portalı
*
* @package EasyAppointments
* @author A.Tselegidis <alextselegidis@gmail.com>
* @copyright Copyright (c) Alex Tselegidis
* @license https://opensource.org/licenses/GPL-3.0 - GPLv3
* @link https://easyappointments.org
* @since v1.0.0
* ---------------------------------------------------------------------------- */
/**
* Appointments model.
*
* @package Models
*/
class Appointments_model extends EA_Model
{
/**
* @var array
*/
protected array $casts = [
'id' => 'integer',
'is_unavailability' => 'boolean',
'id_users_provider' => 'integer',
'id_users_customer' => 'integer',
'id_services' => 'integer',
];
/**
* @var array
*/
protected array $api_resource = [
'id' => 'id',
'book' => 'book_datetime',
'start' => 'start_datetime',
'end' => 'end_datetime',
'location' => 'location',
'color' => 'color',
'status' => 'status',
'notes' => 'notes',
'hash' => 'hash',
'serviceId' => 'id_services',
'providerId' => 'id_users_provider',
'customerId' => 'id_users_customer',
'googleCalendarId' => 'id_google_calendar',
];
/**
* Save (insert or update) an appointment.
*
* @param array $appointment Associative array with the appointment data.
*
* @return int Returns the appointment ID.
*
* @throws InvalidArgumentException
*/
public function save(array $appointment): int
{
$this->validate($appointment);
if (empty($appointment['id'])) {
return $this->insert($appointment);
} else {
return $this->update($appointment);
}
}
/**
* Validate the appointment data.
*
* @param array $appointment Associative array with the appointment data.
*
* @throws InvalidArgumentException
*/
public function validate(array $appointment)
{
// If an appointment ID is provided then check whether the record really exists in the database.
if (!empty($appointment['id'])) {
$count = $this->db->get_where('appointments', ['id' => $appointment['id']])->num_rows();
if (!$count) {
throw new InvalidArgumentException(
'The provided appointment ID does not exist in the database: ' . $appointment['id'],
);
}
}
// Make sure all required fields are provided.
$require_notes = filter_var(setting('require_notes'), FILTER_VALIDATE_BOOLEAN);
if (
empty($appointment['start_datetime']) ||
empty($appointment['end_datetime']) ||
empty($appointment['id_services']) ||
empty($appointment['id_users_provider']) ||
empty($appointment['id_users_customer']) ||
(empty($appointment['notes']) && $require_notes)
) {
throw new InvalidArgumentException('Not all required fields are provided: ' . print_r($appointment, true));
}
// Make sure that the provided appointment date time values are valid.
if (!validate_datetime($appointment['start_datetime'])) {
throw new InvalidArgumentException('The appointment start date time is invalid.');
}
if (!validate_datetime($appointment['end_datetime'])) {
throw new InvalidArgumentException('The appointment end date time is invalid.');
}
// Make the appointment lasts longer than the minimum duration (in minutes).
$diff = (strtotime($appointment['end_datetime']) - strtotime($appointment['start_datetime'])) / 60;
if ($diff < EVENT_MINIMUM_DURATION) {
throw new InvalidArgumentException(
'The appointment duration cannot be less than ' . EVENT_MINIMUM_DURATION . ' minutes.',
);
}
// Make sure the provider ID really exists in the database.
$count = $this->db
->select()
->from('users')
->join('roles', 'roles.id = users.id_roles', 'inner')
->where('users.id', $appointment['id_users_provider'])
->where('roles.slug', DB_SLUG_PROVIDER)
->get()
->num_rows();
if (!$count) {
throw new InvalidArgumentException(
'The appointment provider ID was not found in the database: ' . $appointment['id_users_provider'],
);
}
if (!filter_var($appointment['is_unavailability'], FILTER_VALIDATE_BOOLEAN)) {
// Make sure the customer ID really exists in the database.
$count = $this->db
->select()
->from('users')
->join('roles', 'roles.id = users.id_roles', 'inner')
->where('users.id', $appointment['id_users_customer'])
->where('roles.slug', DB_SLUG_CUSTOMER)
->get()
->num_rows();
if (!$count) {
throw new InvalidArgumentException(
'The appointment customer ID was not found in the database: ' . $appointment['id_users_customer'],
);
}
// Make sure the service ID really exists in the database.
$count = $this->db->get_where('services', ['id' => $appointment['id_services']])->num_rows();
if (!$count) {
throw new InvalidArgumentException('Appointment service id is invalid.');
}
}
}
/**
* Get all appointments that match the provided criteria.
*
* @param array|string|null $where Where conditions.
* @param int|null $limit Record limit.
* @param int|null $offset Record offset.
* @param string|null $order_by Order by.
*
* @return array Returns an array of appointments.
*/
public function get(
array|string $where = null,
int $limit = null,
int $offset = null,
string $order_by = null,
): array {
if ($where !== null) {
$this->db->where($where);
}
if ($order_by) {
$this->db->order_by($order_by);
}
$appointments = $this->db
->get_where('appointments', ['is_unavailability' => false], $limit, $offset)
->result_array();
foreach ($appointments as &$appointment) {
$this->cast($appointment);
}
return $appointments;
}
/**
* Insert a new appointment into the database.
*
* @param array $appointment Associative array with the appointment data.
*
* @return int Returns the appointment ID.
*
* @throws RuntimeException
*/
protected function insert(array $appointment): int
{
$appointment['book_datetime'] = date('Y-m-d H:i:s');
$appointment['create_datetime'] = date('Y-m-d H:i:s');
$appointment['update_datetime'] = date('Y-m-d H:i:s');
$appointment['hash'] = random_string('alnum', 12);
if (!$this->db->insert('appointments', $appointment)) {
throw new RuntimeException('Could not insert appointment.');
}
return $this->db->insert_id();
}
/**
* Update an existing appointment.
*
* @param array $appointment Associative array with the appointment data.
*
* @return int Returns the appointment ID.
*
* @throws RuntimeException
*/
protected function update(array $appointment): int
{
$appointment['update_datetime'] = date('Y-m-d H:i:s');
if (!$this->db->update('appointments', $appointment, ['id' => $appointment['id']])) {
throw new RuntimeException('Could not update appointment record.');
}
return $appointment['id'];
}
/**
* Remove an existing appointment from the database.
*
* @param int $appointment_id Appointment ID.
*
* @throws RuntimeException
*/
public function delete(int $appointment_id): void
{
$this->db->delete('appointments', ['id' => $appointment_id]);
}
/**
* Get a specific appointment from the database.
*
* @param int $appointment_id The ID of the record to be returned.
*
* @return array Returns an array with the appointment data.
*
* @throws InvalidArgumentException
*/
public function find(int $appointment_id): array
{
$appointment = $this->db->get_where('appointments', ['id' => $appointment_id])->row_array();
if (!$appointment) {
throw new InvalidArgumentException(
'The provided appointment ID was not found in the database: ' . $appointment_id,
);
}
$this->cast($appointment);
return $appointment;
}
/**
* Get a specific field value from the database.
*
* @param int $appointment_id Appointment ID.
* @param string $field Name of the value to be returned.
*
* @return mixed Returns the selected appointment value from the database.
*
* @throws InvalidArgumentException
*/
public function value(int $appointment_id, string $field): mixed
{
if (empty($field)) {
throw new InvalidArgumentException('The field argument is cannot be empty.');
}
if (empty($appointment_id)) {
throw new InvalidArgumentException('The appointment ID argument cannot be empty.');
}
// Check whether the appointment exists.
$query = $this->db->get_where('appointments', ['id' => $appointment_id]);
if (!$query->num_rows()) {
throw new InvalidArgumentException(
'The provided appointment ID was not found in the database: ' . $appointment_id,
);
}
// Check if the required field is part of the appointment data.
$appointment = $query->row_array();
$this->cast($appointment);
if (!array_key_exists($field, $appointment)) {
throw new InvalidArgumentException('The requested field was not found in the appointment data: ' . $field);
}
return $appointment[$field];
}
/**
* Remove all the Google Calendar event IDs from appointment records.
*
* @param int $provider_id Matching provider ID.
*/
public function clear_google_sync_ids(int $provider_id)
{
$this->db->update('appointments', ['id_google_calendar' => null], ['id_users_provider' => $provider_id]);
}
/**
* Get the attendants number for the requested period.
*
* @param DateTime $start Period start.
* @param DateTime $end Period end.
* @param int $service_id Service ID.
* @param int $provider_id Provider ID.
* @param int|null $exclude_appointment_id Exclude an appointment from the result set.
*
* @return int Returns the number of appointments that match the provided criteria.
*/
public function get_attendants_number_for_period(
DateTime $start,
DateTime $end,
int $service_id,
int $provider_id,
int $exclude_appointment_id = null,
): int {
if ($exclude_appointment_id) {
$this->db->where('id !=', $exclude_appointment_id);
}
$result = $this->db
->select('count(*) AS attendants_number')
->from('appointments')
->group_start()
->group_start()
->where('start_datetime <=', $start->format('Y-m-d H:i:s'))
->where('end_datetime >', $start->format('Y-m-d H:i:s'))
->group_end()
->or_group_start()
->where('start_datetime <', $end->format('Y-m-d H:i:s'))
->where('end_datetime >=', $end->format('Y-m-d H:i:s'))
->group_end()
->group_end()
->where('id_services', $service_id)
->where('id_users_provider', $provider_id)
->get()
->row_array();
return $result['attendants_number'];
}
/**
*
* Returns the number of the other service attendants number for the provided time slot.
*
* @param DateTime $start Period start.
* @param DateTime $end Period end.
* @param int $service_id Service ID.
* @param int $provider_id Provider ID.
* @param int|null $exclude_appointment_id Exclude an appointment from the result set.
*
* @return int Returns the number of appointments that match the provided criteria.
*/
public function get_other_service_attendants_number(
DateTime $start,
DateTime $end,
int $service_id,
int $provider_id,
int $exclude_appointment_id = null,
): int {
if ($exclude_appointment_id) {
$this->db->where('id !=', $exclude_appointment_id);
}
$result = $this->db
->select('count(*) AS attendants_number')
->from('appointments')
->group_start()
->group_start()
->where('start_datetime <=', $start->format('Y-m-d H:i:s'))
->where('end_datetime >', $start->format('Y-m-d H:i:s'))
->group_end()
->or_group_start()
->where('start_datetime <', $end->format('Y-m-d H:i:s'))
->where('end_datetime >=', $end->format('Y-m-d H:i:s'))
->group_end()
->group_end()
->where('id_services !=', $service_id)
->where('id_users_provider', $provider_id)
->get()
->row_array();
return $result['attendants_number'];
}
/**
* Get the query builder interface, configured for use with the appointments table.
*
* @return CI_DB_query_builder
*/
public function query(): CI_DB_query_builder
{
return $this->db->from('appointments');
}
/**
* Search appointments by the provided keyword.
*
* @param string $keyword Search keyword.
* @param int|null $limit Record limit.
* @param int|null $offset Record offset.
* @param string|null $order_by Order by.
*
* @return array Returns an array of appointments.
*/
public function search(string $keyword, int $limit = null, int $offset = null, string $order_by = null): array
{
$appointments = $this->db
->select('appointments.*')
->from('appointments')
->join('services', 'services.id = appointments.id_services', 'left')
->join('users AS providers', 'providers.id = appointments.id_users_provider', 'inner')
->join('users AS customers', 'customers.id = appointments.id_users_customer', 'left')
->where('is_unavailability', false)
->group_start()
->like('appointments.start_datetime', $keyword)
->or_like('appointments.end_datetime', $keyword)
->or_like('appointments.location', $keyword)
->or_like('appointments.hash', $keyword)
->or_like('appointments.notes', $keyword)
->or_like('services.name', $keyword)
->or_like('services.description', $keyword)
->or_like('providers.first_name', $keyword)
->or_like('providers.last_name', $keyword)
->or_like('providers.email', $keyword)
->or_like('providers.phone_number', $keyword)
->or_like('customers.first_name', $keyword)
->or_like('customers.last_name', $keyword)
->or_like('customers.email', $keyword)
->or_like('customers.phone_number', $keyword)
->group_end()
->limit($limit)
->offset($offset)
->order_by($order_by)
->get()
->result_array();
foreach ($appointments as &$appointment) {
$this->cast($appointment);
}
return $appointments;
}
/**
* Load related resources to an appointment.
*
* @param array $appointment Associative array with the appointment data.
* @param array $resources Resource names to be attached ("service", "provider", "customer" supported).
*
* @throws InvalidArgumentException
*/
public function load(array &$appointment, array $resources)
{
if (empty($appointment) || empty($resources)) {
return;
}
foreach ($resources as $resource) {
switch ($resource) {
case 'service':
$appointment['service'] = $this->db
->get_where('services', [
'id' => $appointment['id_services'] ?? ($appointment['serviceId'] ?? null),
])
->row_array();
break;
case 'provider':
$appointment['provider'] = $this->db
->get_where('users', [
'id' => $appointment['id_users_provider'] ?? ($appointment['providerId'] ?? null),
])
->row_array();
break;
case 'customer':
$appointment['customer'] = $this->db
->get_where('users', [
'id' => $appointment['id_users_customer'] ?? ($appointment['customerId'] ?? null),
])
->row_array();
break;
default:
throw new InvalidArgumentException(
'The requested appointment relation is not supported: ' . $resource,
);
}
}
}
/**
* Convert the database appointment record to the equivalent API resource.
*
* @param array $appointment Appointment data.
*/
public function api_encode(array &$appointment)
{
$encoded_resource = [
'id' => array_key_exists('id', $appointment) ? (int) $appointment['id'] : null,
'book' => $appointment['book_datetime'],
'start' => $appointment['start_datetime'],
'end' => $appointment['end_datetime'],
'hash' => $appointment['hash'],
'color' => $appointment['color'],
'status' => $appointment['status'],
'location' => $appointment['location'],
'notes' => $appointment['notes'],
'customerId' => $appointment['id_users_customer'] !== null ? (int) $appointment['id_users_customer'] : null,
'providerId' => $appointment['id_users_provider'] !== null ? (int) $appointment['id_users_provider'] : null,
'serviceId' => $appointment['id_services'] !== null ? (int) $appointment['id_services'] : null,
'googleCalendarId' =>
$appointment['id_google_calendar'] !== null ? (int) $appointment['id_google_calendar'] : null,
];
$appointment = $encoded_resource;
}
/**
* Convert the API resource to the equivalent database appointment record.
*
* @param array $appointment API resource.
* @param array|null $base Base appointment data to be overwritten with the provided values (useful for updates).
*/
public function api_decode(array &$appointment, array $base = null)
{
$decoded_request = $base ?: [];
if (array_key_exists('id', $appointment)) {
$decoded_request['id'] = $appointment['id'];
}
if (array_key_exists('book', $appointment)) {
$decoded_request['book_datetime'] = $appointment['book'];
}
if (array_key_exists('start', $appointment)) {
$decoded_request['start_datetime'] = $appointment['start'];
}
if (array_key_exists('end', $appointment)) {
$decoded_request['end_datetime'] = $appointment['end'];
}
if (array_key_exists('hash', $appointment)) {
$decoded_request['hash'] = $appointment['hash'];
}
if (array_key_exists('location', $appointment)) {
$decoded_request['location'] = $appointment['location'];
}
if (array_key_exists('notes', $appointment)) {
$decoded_request['notes'] = $appointment['notes'];
}
if (array_key_exists('customerId', $appointment)) {
$decoded_request['id_users_customer'] = $appointment['customerId'];
}
if (array_key_exists('providerId', $appointment)) {
$decoded_request['id_users_provider'] = $appointment['providerId'];
}
if (array_key_exists('serviceId', $appointment)) {
$decoded_request['id_services'] = $appointment['serviceId'];
}
if (array_key_exists('googleCalendarId', $appointment)) {
$decoded_request['id_google_calendar'] = $appointment['googleCalendarId'];
}
$decoded_request['is_unavailability'] = false;
$appointment = $decoded_request;
}
}