File: /home/slfopp7cb1df/public_html/sitepacket.com/system/app/Libraries/Cron_job.php
<?php
namespace App\Libraries;
use App\Controllers\App_Controller;
use App\Libraries\Google_calendar_events;
use App\Libraries\Imap;
use App\Libraries\Outlook_imap;
use App\Libraries\Reminders;
class Cron_job {
private $today = null;
private $current_time = null;
private $ci = null;
function run() {
$this->today = get_today_date();
$this->ci = new App_Controller();
$this->current_time = strtotime(get_current_utc_time());
$this->call_hourly_jobs();
$this->call_daily_jobs();
try {
$this->run_imap();
} catch (\Exception $e) {
echo $e;
}
try {
$this->get_google_calendar_events();
} catch (\Exception $e) {
echo $e;
}
try {
$this->close_inactive_tickets();
} catch (\Exception $e) {
echo $e;
}
}
private function call_hourly_jobs() {
//wait 1 hour for each call of following actions
if ($this->_is_hourly_job_runnable()) {
try {
$this->create_recurring_invoices();
} catch (\Exception $e) {
echo $e;
}
try {
$this->create_recurring_expenses();
} catch (\Exception $e) {
echo $e;
}
try {
$this->send_invoice_due_pre_reminder();
} catch (\Exception $e) {
echo $e;
}
try {
$this->send_invoice_due_after_reminder();
} catch (\Exception $e) {
echo $e;
}
try {
$this->send_recurring_invoice_creation_reminder();
} catch (\Exception $e) {
echo $e;
}
try {
$this->create_recurring_tasks();
} catch (\Exception $e) {
echo $e;
}
try {
$this->send_task_reminder_notifications();
} catch (\Exception $e) {
echo $e;
}
try {
$this->create_recurring_reminders();
} catch (\Exception $e) {
echo $e;
}
try {
$this->create_subscription_invoices();
} catch (\Exception $e) {
echo $e;
}
try {
$this->_create_subscription_reminders();
} catch (\Exception $e) {
echo $e;
}
try {
$this->_send_available_reminders();
} catch (\Exception $e) {
echo $e;
}
$this->ci->Settings_model->save_setting("last_hourly_job_time", $this->current_time);
}
}
private function _is_hourly_job_runnable() {
$last_hourly_job_time = get_setting('last_hourly_job_time');
if (!$last_hourly_job_time || ($this->current_time > ($last_hourly_job_time * 1 + 3600))) {
return true;
}
}
private function send_invoice_due_pre_reminder() {
$reminder_date = get_setting("send_invoice_due_pre_reminder");
$reminder_date2 = get_setting("send_invoice_due_pre_second_reminder");
if (!($reminder_date || $reminder_date2)) {
return false;
}
//prepare invoice due date accroding to the setting
$reminder_due_date = $reminder_date ? add_period_to_date($this->today, $reminder_date, "days") : "";
$reminder_due_date2 = $reminder_date2 ? add_period_to_date($this->today, $reminder_date2, "days") : "";
$invoices = $this->ci->Invoices_model->get_details(array(
"status" => "not_paid_and_partially_paid", //find all invoices which are not paid yet but due date not expired
"reminder_due_date" => $reminder_due_date,
"reminder_due_date2" => $reminder_due_date2,
"exclude_due_reminder_date" => $this->today //don't find invoices which reminder already sent today
))->getResult();
foreach ($invoices as $invoice) {
log_notification("invoice_due_reminder_before_due_date", array("invoice_id" => $invoice->id), "0");
}
}
private function send_invoice_due_after_reminder() {
$reminder_date = get_setting("send_invoice_due_after_reminder");
$reminder_date2 = get_setting("send_invoice_due_after_second_reminder");
if (!($reminder_date || $reminder_date2)) {
return false;
}
//prepare invoice due date accroding to the setting
$reminder_due_date = $reminder_date ? subtract_period_from_date($this->today, $reminder_date, "days") : "";
$reminder_due_date2 = $reminder_date2 ? subtract_period_from_date($this->today, $reminder_date2, "days") : "";
$invoices = $this->ci->Invoices_model->get_details(array(
"status" => "overdue", //find all invoices where due date has expired
"reminder_due_date" => $reminder_due_date,
"reminder_due_date2" => $reminder_due_date2,
"exclude_due_reminder_date" => $this->today //don't find invoices which reminder already sent today
))->getResult();
foreach ($invoices as $invoice) {
log_notification("invoice_overdue_reminder", array("invoice_id" => $invoice->id), "0");
}
}
private function send_recurring_invoice_creation_reminder() {
$reminder_date = get_setting("send_recurring_invoice_reminder_before_creation");
if ($reminder_date) {
//prepare invoice due date accroding to the setting
$start_date = add_period_to_date($this->today, get_setting("send_recurring_invoice_reminder_before_creation"), "days");
$invoices = $this->ci->Invoices_model->get_details(array(
"status" => "not_paid", //non-draft invoices
"recurring" => 1,
"next_recurring_start_date" => $start_date,
"next_recurring_end_date" => $start_date, //both should be same
"exclude_recurring_reminder_date" => $this->today //don't find invoices which reminder already sent today
))->getResult();
foreach ($invoices as $invoice) {
log_notification("recurring_invoice_creation_reminder", array("invoice_id" => $invoice->id), "0");
}
}
}
private function create_recurring_invoices() {
$recurring_invoices = $this->ci->Invoices_model->get_renewable_invoices($this->today);
if ($recurring_invoices->resultID->num_rows) {
foreach ($recurring_invoices->getResult() as $invoice) {
$this->_create_new_invoice($invoice);
}
}
}
//create new invoice from a recurring invoice
private function _create_new_invoice($invoice) {
//don't update the next recurring date when updating invoice manually?
//stop backdated recurring invoice creation.
//check recurring invoice once/hour?
//settings: send invoice to client
$bill_date = $invoice->next_recurring_date;
$diff_of_due_date = get_date_difference_in_days($invoice->due_date, $invoice->bill_date); //calculate the due date difference of the original invoice
$due_date = add_period_to_date($bill_date, $diff_of_due_date, "days");
$_new_invoice_data = array(
"client_id" => $invoice->client_id,
"project_id" => $invoice->project_id,
"bill_date" => $bill_date,
"due_date" => $due_date,
"note" => $invoice->note,
"status" => "draft",
"tax_id" => $invoice->tax_id,
"tax_id2" => $invoice->tax_id2,
"tax_id3" => $invoice->tax_id3,
"recurring_invoice_id" => $invoice->id,
"discount_amount" => $invoice->discount_amount,
"discount_amount_type" => $invoice->discount_amount_type,
"discount_type" => $invoice->discount_type,
"company_id" => $invoice->company_id,
"invoice_subtotal" => $invoice->invoice_subtotal,
"invoice_total" => $invoice->invoice_total,
"discount_total" => $invoice->discount_total,
"tax" => $invoice->tax,
"tax2" => $invoice->tax2,
"tax3" => $invoice->tax3
);
$new_invoice_data = array_merge($_new_invoice_data, prepare_invoice_display_id_data($due_date, $bill_date));
//create new invoice
$new_invoice_id = $this->ci->Invoices_model->ci_save($new_invoice_data);
//create invoice items
$items = $this->ci->Invoice_items_model->get_details(array("invoice_id" => $invoice->id))->getResult();
foreach ($items as $item) {
//create invoice items for new invoice
$new_invoice_item_data = array(
"title" => $item->title,
"description" => $item->description,
"quantity" => $item->quantity,
"unit_type" => $item->unit_type,
"rate" => $item->rate,
"total" => $item->total,
"taxable" => $item->taxable,
"invoice_id" => $new_invoice_id,
"item_id" => $item->item_id
);
$this->ci->Invoice_items_model->ci_save($new_invoice_item_data);
}
//update the main recurring invoice
$no_of_cycles_completed = $invoice->no_of_cycles_completed + 1;
$next_recurring_date = add_period_to_date($bill_date, $invoice->repeat_every, $invoice->repeat_type);
$recurring_invoice_data = array(
"next_recurring_date" => $next_recurring_date,
"no_of_cycles_completed" => $no_of_cycles_completed
);
$this->ci->Invoices_model->ci_save($recurring_invoice_data, $invoice->id);
//finally send notification
log_notification("recurring_invoice_created_vai_cron_job", array("invoice_id" => $new_invoice_id), "0");
}
private function create_subscription_invoices() {
$subscriptions = $this->ci->Subscriptions_model->get_renewable_subscriptions($this->today);
if ($subscriptions->resultID->num_rows) {
foreach ($subscriptions->getResult() as $subscription) {
$this->_create_new_invoice_of_subscription($subscription);
}
}
}
//create new invoice from a subscription
private function _create_new_invoice_of_subscription($subscription_info) {
$invoice_id = create_invoice_from_subscription($subscription_info->id);
//update the main recurring subscription
$no_of_cycles_completed = $subscription_info->no_of_cycles_completed + 1;
$next_recurring_date = add_period_to_date($subscription_info->next_recurring_date, $subscription_info->repeat_every, $subscription_info->repeat_type);
$subscription_data = array(
"next_recurring_date" => $next_recurring_date,
"no_of_cycles_completed" => $no_of_cycles_completed
);
$this->ci->Subscriptions_model->ci_save($subscription_data, $subscription_info->id);
//finally send notification
log_notification("subscription_invoice_created_via_cron_job", array("invoice_id" => $invoice_id, "subscription_id" => $subscription_info->id), "0");
}
private function get_google_calendar_events() {
$Google_calendar_events = new Google_calendar_events();
$Google_calendar_events->get_google_calendar_events();
}
private function run_imap() {
if (!$this->_is_imap_callable()) {
return false;
}
if (!get_setting('imap_type') || get_setting('imap_type') === "general_imap") {
$imap = new Imap();
$imap->run_imap();
} else if (get_setting('imap_type') === "microsoft_outlook") {
$imap = new Outlook_imap();
$imap->run_imap();
} else if (get_setting('imap_type') === "gmail_imap") {
$imap = new Gmail_imap();
$imap->run_imap();
}
}
private function _is_imap_callable() {
//check if settings is enabled and authorized
if (!(get_setting("enable_email_piping") && get_setting("imap_authorized"))) {
return false;
}
return true;
}
private function create_recurring_tasks() {
if (!get_setting("enable_recurring_option_for_tasks")) {
return false;
}
$date = $this->today;
//if create recurring task before certain days setting is active,
//add the days with today
$create_recurring_tasks_before = get_setting("create_recurring_tasks_before");
if ($create_recurring_tasks_before) {
$date = add_period_to_date($date, $create_recurring_tasks_before, "days");
}
$recurring_tasks = $this->ci->Tasks_model->get_renewable_tasks($date);
if ($recurring_tasks->resultID->num_rows) {
foreach ($recurring_tasks->getResult() as $task) {
$this->_create_new_task($task);
}
}
}
//create new task from a recurring task
private function _create_new_task($task) {
//don't update the next recurring date when updating task manually
//stop backdated recurring task creation.
//check recurring task once/hour?
$start_date = $task->next_recurring_date;
$deadline = NULL;
$context = $task->context;
if ($task->deadline) {
$task_start_date = $task->start_date ? $task->start_date : $task->created_date;
$diff_of_deadline = get_date_difference_in_days($task->deadline, $task_start_date); //calculate the deadline difference of the original task
$deadline = add_period_to_date($start_date, $diff_of_deadline, "days");
}
$new_task_data = array(
"title" => $task->title,
"description" => $task->description,
"project_id" => $task->project_id,
"milestone_id" => $task->milestone_id,
"points" => $task->points,
"status_id" => 1, //new tasks should be on ToDo
"context" => $context,
"client_id" => $task->client_id,
"lead_id" => $task->lead_id,
"invoice_id" => $task->invoice_id,
"estimate_id" => $task->estimate_id,
"order_id" => $task->order_id,
"contract_id" => $task->contract_id,
"proposal_id" => $task->proposal_id,
"expense_id" => $task->expense_id,
"subscription_id" => $task->subscription_id,
"priority_id" => $task->priority_id,
"labels" => $task->labels,
"start_date" => $start_date,
"deadline" => $deadline,
"recurring_task_id" => $task->id,
"assigned_to" => $task->assigned_to,
"collaborators" => $task->collaborators,
"created_date" => get_current_utc_time(),
"activity_log_created_by_app" => true,
"created_by" => $task->created_by
);
$new_task_data["sort"] = $this->ci->Tasks_model->get_next_sort_value($task->project_id, $new_task_data["status_id"]);
//create new task
$new_task_id = $this->ci->Tasks_model->ci_save($new_task_data);
//save custom fields
$custom_fields = $this->ci->Custom_fields_model->get_combined_details("tasks", $task->id, 1, "staff")->getResult();
foreach ($custom_fields as $field) {
$field_value_data = array(
"related_to_type" => "tasks",
"related_to_id" => $new_task_id,
"custom_field_id" => $field->id,
"value" => $field->value
);
$field_value_data = clean_data($field_value_data);
$this->ci->Custom_field_values_model->ci_save($field_value_data);
}
//create checklist items
$Checklist_items_model = model("App\Models\Checklist_items_model");
$checklist_item_options = array("task_id" => $task->id);
$checklist_items = $Checklist_items_model->get_details($checklist_item_options);
if ($checklist_items->resultID->num_rows) {
foreach ($checklist_items->getResult() as $item) {
$checklist_item_data = array(
"title" => $item->title,
"is_checked" => $item->is_checked,
"task_id" => $new_task_id,
"sort" => $item->sort
);
$Checklist_items_model->ci_save($checklist_item_data);
}
}
//create sub tasks
$sub_tasks = $this->ci->Tasks_model->get_all_where(array("parent_task_id" => $task->id, "deleted" => 0))->getResult();
foreach ($sub_tasks as $sub_task) {
//prepare new sub task data
$sub_task_data = (array) $sub_task;
unset($sub_task_data["id"]);
unset($sub_task_data["blocked_by"]);
unset($sub_task_data["blocking"]);
if ($task->start_date && $sub_task->start_date) {
$sub_task_data['start_date'] = $start_date;
} else {
$sub_task_data['start_date'] = NULL;
}
$sub_task_data['status_id'] = 1;
$sub_task_data['parent_task_id'] = $new_task_id;
$sub_task_data['created_date'] = get_current_utc_time();
$sub_task_data['deadline'] = NULL;
$sub_task_data["sort"] = $this->ci->Tasks_model->get_next_sort_value(get_array_value($sub_task_data, "project_id"), $sub_task_data["status_id"]);
$sub_task_save_id = $this->ci->Tasks_model->ci_save($sub_task_data);
//create sub tasks checklist
$checklist_items = $Checklist_items_model->get_all_where(array("task_id" => $sub_task->id, "deleted" => 0))->getResult();
foreach ($checklist_items as $checklist_item) {
//prepare new checklist data
$checklist_item_data = (array) $checklist_item;
unset($checklist_item_data["id"]);
$checklist_item_data['task_id'] = $sub_task_save_id;
$Checklist_items_model->ci_save($checklist_item_data);
}
}
//update the main recurring task
$no_of_cycles_completed = $task->no_of_cycles_completed + 1;
$next_recurring_date = add_period_to_date($start_date, $task->repeat_every, $task->repeat_type);
$recurring_task_data = array(
"next_recurring_date" => $next_recurring_date,
"no_of_cycles_completed" => $no_of_cycles_completed
);
$this->ci->Tasks_model->save_reminder_date($recurring_task_data, $task->id);
//send notification
if ($context === "project") {
$notification_option = array("project_id" => $task->project_id, "task_id" => $new_task_id);
} else if ($context === "general") {
$notification_option = array("task_id" => $new_task_id);
} else {
$context_id_key = $context . "_id";
$context_id_value = $task->{$context . "_id"};
$notification_option = array("$context_id_key" => $context_id_value, "task_id" => $new_task_id);
}
log_notification("recurring_task_created_via_cron_job", $notification_option, "0");
}
private function send_task_reminder_notifications() {
$notification_option = array("notification_multiple_tasks" => true);
log_notification("project_task_deadline_pre_reminder", $notification_option, "0");
log_notification("project_task_deadline_overdue_reminder", $notification_option, "0");
log_notification("project_task_reminder_on_the_day_of_deadline", $notification_option, "0");
}
private function close_inactive_tickets() {
$inactive_ticket_closing_date = get_setting("inactive_ticket_closing_date");
if (!(!$inactive_ticket_closing_date || ($inactive_ticket_closing_date != $this->today))) {
return false;
}
$auto_close_ticket_after_days = get_setting("auto_close_ticket_after");
if ($auto_close_ticket_after_days) {
//prepare last activity date accroding to the setting
$last_activity_date = subtract_period_from_date($this->today, get_setting("auto_close_ticket_after"), "days");
$tickets = $this->ci->Tickets_model->get_details(array(
"status" => "open", //don't find closed tickets
"last_activity_date_or_before" => $last_activity_date
))->getResult();
foreach ($tickets as $ticket) {
//make ticket closed
$ticket_data = array(
"status" => "closed",
"closed_at" => get_current_utc_time()
);
$this->ci->Tickets_model->ci_save($ticket_data, $ticket->id);
//send notification
log_notification("ticket_closed", array("ticket_id" => $ticket->id), "0");
}
}
$this->ci->Settings_model->save_setting("inactive_ticket_closing_date", $this->today);
}
private function create_recurring_expenses() {
$recurring_expenses = $this->ci->Expenses_model->get_renewable_expenses($this->today);
if ($recurring_expenses->resultID->num_rows) {
foreach ($recurring_expenses->getResult() as $expense) {
$this->_create_new_expense($expense);
}
}
}
//create new expense from a recurring expense
private function _create_new_expense($expense) {
//don't update the next recurring date when updating expense manually?
//stop backdated recurring expense creation.
//check recurring expense once/hour?
$expense_date = $expense->next_recurring_date;
$new_expense_data = array(
"title" => $expense->title,
"expense_date" => $expense_date,
"description" => $expense->description,
"category_id" => $expense->category_id,
"amount" => $expense->amount,
"project_id" => $expense->project_id,
"user_id" => $expense->user_id,
"tax_id" => $expense->tax_id,
"tax_id2" => $expense->tax_id2,
"recurring_expense_id" => $expense->id
);
//create new expense
$new_expense_id = $this->ci->Expenses_model->ci_save($new_expense_data);
//update the main recurring expense
$no_of_cycles_completed = $expense->no_of_cycles_completed + 1;
$next_recurring_date = add_period_to_date($expense_date, $expense->repeat_every, $expense->repeat_type);
$recurring_expense_data = array(
"next_recurring_date" => $next_recurring_date,
"no_of_cycles_completed" => $no_of_cycles_completed
);
$this->ci->Expenses_model->ci_save($recurring_expense_data, $expense->id);
//finally send notification
//log_notification("recurring_expense_created_vai_cron_job", array("expense_id" => $new_expense_id), "0");
}
private function create_recurring_reminders() {
$options = array(
"type" => "all",
"recurring" => true,
"reminder_status" => "new",
);
$recurring_reminders = $this->ci->Events_model->get_details($options)->getResult();
foreach ($recurring_reminders as $reminder) {
$now = get_my_local_time();
$target_time = is_null($reminder->next_recurring_time) ? ($reminder->start_date . " " . $reminder->start_time) : $reminder->next_recurring_time;
if ($target_time < $now && (!$reminder->no_of_cycles || $reminder->no_of_cycles_completed < $reminder->no_of_cycles)) {
$data["next_recurring_time"] = add_period_to_date($target_time, $reminder->repeat_every, $reminder->repeat_type, "Y-m-d H:i:s");
$data['no_of_cycles_completed'] = (int) $reminder->no_of_cycles_completed + 1;
$this->ci->Events_model->ci_save($data, $reminder->id);
}
}
}
private function call_daily_jobs() {
//wait 1 day for each call of following actions
if ($this->_is_daily_job_runnable()) {
try {
$this->remove_old_session_data();
} catch (\Exception $e) {
echo $e;
}
$this->ci->Settings_model->save_setting("last_daily_cron_job_date", $this->today);
}
}
private function _is_daily_job_runnable() {
$last_daily_cron_job_date = get_setting('last_daily_cron_job_date');
if (!$last_daily_cron_job_date || ($this->today > $last_daily_cron_job_date)) {
return true;
}
}
private function remove_old_session_data() {
$Ci_sessions_model = model('App\Models\Ci_sessions_model');
$last_weak_date = subtract_period_from_date($this->today, 7, "days");
$Ci_sessions_model->delete_session_by_date($last_weak_date);
}
private function _create_subscription_reminders() {
$reminders = new Reminders();
$reminders->create_reminders("subscription");
}
private function _send_available_reminders() {
$reminders = new Reminders();
$reminders->send_available_reminders();
}
}