123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411 |
- <?php
- // Copyright (C) 2010-2012 Combodo SARL
- //
- // This file is part of iTop.
- //
- // iTop is free software; you can redistribute it and/or modify
- // it under the terms of the GNU Affero General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // iTop is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU Affero General Public License for more details.
- //
- // You should have received a copy of the GNU Affero General Public License
- // along with iTop. If not, see <http://www.gnu.org/licenses/>
- /**
- * Persistent classes (internal): user defined actions
- *
- * @copyright Copyright (C) 2010-2012 Combodo SARL
- * @license http://opensource.org/licenses/AGPL-3.0
- */
- require_once(APPROOT.'/core/asynctask.class.inc.php');
- require_once(APPROOT.'/core/email.class.inc.php');
- /**
- * A user defined action, to customize the application
- *
- * @package iTopORM
- */
- abstract class Action extends cmdbAbstractObject
- {
- public static function Init()
- {
- $aParams = array
- (
- "category" => "core/cmdb",
- "key_type" => "autoincrement",
- "name_attcode" => "name",
- "state_attcode" => "",
- "reconc_keys" => array('name'),
- "db_table" => "priv_action",
- "db_key_field" => "id",
- "db_finalclass_field" => "realclass",
- "display_template" => "",
- );
- MetaModel::Init_Params($aParams);
- //MetaModel::Init_InheritAttributes();
- MetaModel::Init_AddAttribute(new AttributeString("name", array("allowed_values"=>null, "sql"=>"name", "default_value"=>null, "is_null_allowed"=>false, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeString("description", array("allowed_values"=>null, "sql"=>"description", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeEnum("status", array("allowed_values"=>new ValueSetEnum(array('test'=>'Being tested' ,'enabled'=>'In production', 'disabled'=>'Inactive')), "sql"=>"status", "default_value"=>"test", "is_null_allowed"=>false, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeLinkedSetIndirect("trigger_list", array("linked_class"=>"lnkTriggerAction", "ext_key_to_me"=>"action_id", "ext_key_to_remote"=>"trigger_id", "allowed_values"=>null, "count_min"=>0, "count_max"=>0, "depends_on"=>array())));
- // Display lists
- MetaModel::Init_SetZListItems('details', array('name', 'description', 'status', 'trigger_list')); // Attributes to be displayed for the complete details
- MetaModel::Init_SetZListItems('list', array('finalclass', 'name', 'description', 'status')); // Attributes to be displayed for a list
- // Search criteria
- // MetaModel::Init_SetZListItems('standard_search', array('name')); // Criteria of the std search form
- // MetaModel::Init_SetZListItems('advanced_search', array('name')); // Criteria of the advanced search form
- }
- abstract public function DoExecute($oTrigger, $aContextArgs);
- public function IsActive()
- {
- switch($this->Get('status'))
- {
- case 'enabled':
- case 'test':
- return true;
- default:
- return false;
- }
- }
- public function IsBeingTested()
- {
- switch($this->Get('status'))
- {
- case 'test':
- return true;
- default:
- return false;
- }
- }
- }
- /**
- * A notification
- *
- * @package iTopORM
- */
- abstract class ActionNotification extends Action
- {
- public static function Init()
- {
- $aParams = array
- (
- "category" => "core/cmdb",
- "key_type" => "autoincrement",
- "name_attcode" => "name",
- "state_attcode" => "",
- "reconc_keys" => array('name'),
- "db_table" => "priv_action_notification",
- "db_key_field" => "id",
- "db_finalclass_field" => "",
- "display_template" => "",
- );
- MetaModel::Init_Params($aParams);
- MetaModel::Init_InheritAttributes();
- // Display lists
- MetaModel::Init_SetZListItems('details', array('name', 'description', 'status', 'trigger_list')); // Attributes to be displayed for the complete details
- MetaModel::Init_SetZListItems('list', array('finalclass', 'name', 'description', 'status')); // Attributes to be displayed for a list
- // Search criteria
- // MetaModel::Init_SetZListItems('standard_search', array('name')); // Criteria of the std search form
- // MetaModel::Init_SetZListItems('advanced_search', array('name')); // Criteria of the advanced search form
- }
- }
- /**
- * An email notification
- *
- * @package iTopORM
- */
- class ActionEmail extends ActionNotification
- {
- public static function Init()
- {
- $aParams = array
- (
- "category" => "core/cmdb,application",
- "key_type" => "autoincrement",
- "name_attcode" => "name",
- "state_attcode" => "",
- "reconc_keys" => array('name'),
- "db_table" => "priv_action_email",
- "db_key_field" => "id",
- "db_finalclass_field" => "",
- "display_template" => "",
- );
- MetaModel::Init_Params($aParams);
- MetaModel::Init_InheritAttributes();
- MetaModel::Init_AddAttribute(new AttributeEmailAddress("test_recipient", array("allowed_values"=>null, "sql"=>"test_recipient", "default_value"=>"", "is_null_allowed"=>true, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeString("from", array("allowed_values"=>null, "sql"=>"from", "default_value"=>null, "is_null_allowed"=>false, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeString("reply_to", array("allowed_values"=>null, "sql"=>"reply_to", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeOQL("to", array("allowed_values"=>null, "sql"=>"to", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeOQL("cc", array("allowed_values"=>null, "sql"=>"cc", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeOQL("bcc", array("allowed_values"=>null, "sql"=>"bcc", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeTemplateString("subject", array("allowed_values"=>null, "sql"=>"subject", "default_value"=>null, "is_null_allowed"=>false, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeTemplateText("body", array("allowed_values"=>null, "sql"=>"body", "default_value"=>null, "is_null_allowed"=>false, "depends_on"=>array())));
- MetaModel::Init_AddAttribute(new AttributeEnum("importance", array("allowed_values"=>new ValueSetEnum('low,normal,high'), "sql"=>"importance", "default_value"=>'normal', "is_null_allowed"=>false, "depends_on"=>array())));
- // Display lists
- MetaModel::Init_SetZListItems('details', array('name', 'description', 'status', 'test_recipient', 'from', 'reply_to', 'to', 'cc', 'bcc', 'subject', 'body', 'importance', 'trigger_list')); // Attributes to be displayed for the complete details
- MetaModel::Init_SetZListItems('list', array('name', 'status', 'to', 'subject')); // Attributes to be displayed for a list
- // Search criteria
- MetaModel::Init_SetZListItems('standard_search', array('name','description', 'status', 'subject')); // Criteria of the std search form
- // MetaModel::Init_SetZListItems('advanced_search', array('name')); // Criteria of the advanced search form
- }
- // count the recipients found
- protected $m_iRecipients;
- // Errors management : not that simple because we need that function to be
- // executed in the background, while making sure that any issue would be reported clearly
- protected $m_aMailErrors; //array of strings explaining the issue
- // returns a the list of emails as a string, or a detailed error description
- protected function FindRecipients($sRecipAttCode, $aArgs)
- {
- $sOQL = $this->Get($sRecipAttCode);
- if (strlen($sOQL) == '') return '';
- try
- {
- $oSearch = DBObjectSearch::FromOQL($sOQL);
- $oSearch->AllowAllData();
- }
- catch (OQLException $e)
- {
- $this->m_aMailErrors[] = "query syntax error for recipient '$sRecipAttCode'";
- return $e->getMessage();
- }
- $sClass = $oSearch->GetClass();
- // Determine the email attribute (the first one will be our choice)
- foreach (MetaModel::ListAttributeDefs($sClass) as $sAttCode => $oAttDef)
- {
- if ($oAttDef instanceof AttributeEmailAddress)
- {
- $sEmailAttCode = $sAttCode;
- // we've got one, exit the loop
- break;
- }
- }
- if (!isset($sEmailAttCode))
- {
- $this->m_aMailErrors[] = "wrong target for recipient '$sRecipAttCode'";
- return "The objects of the class '$sClass' do not have any email attribute";
- }
- $oSet = new DBObjectSet($oSearch, array() /* order */, $aArgs);
- $aRecipients = array();
- while ($oObj = $oSet->Fetch())
- {
- $sAddress = trim($oObj->Get($sEmailAttCode));
- if (strlen($sAddress) > 0)
- {
- $aRecipients[] = $sAddress;
- $this->m_iRecipients++;
- }
- }
- return implode(', ', $aRecipients);
- }
- public function DoExecute($oTrigger, $aContextArgs)
- {
- if (MetaModel::IsLogEnabledNotification())
- {
- $oLog = new EventNotificationEmail();
- if ($this->IsBeingTested())
- {
- $oLog->Set('message', 'TEST - Notification sent ('.$this->Get('test_recipient').')');
- }
- else
- {
- $oLog->Set('message', 'Notification pending');
- }
- $oLog->Set('userinfo', UserRights::GetUser());
- $oLog->Set('trigger_id', $oTrigger->GetKey());
- $oLog->Set('action_id', $this->GetKey());
- $oLog->Set('object_id', $aContextArgs['this->object()']->GetKey());
- // Must be inserted now so that it gets a valid id that will make the link
- // between an eventual asynchronous task (queued) and the log
- $oLog->DBInsertNoReload();
- }
- else
- {
- $oLog = null;
- }
- try
- {
- $sRes = $this->_DoExecute($oTrigger, $aContextArgs, $oLog);
- if ($this->IsBeingTested())
- {
- $sPrefix = 'TEST ('.$this->Get('test_recipient').') - ';
- }
- else
- {
- $sPrefix = '';
- }
- $oLog->Set('message', $sPrefix.$sRes);
- }
- catch (Exception $e)
- {
- if ($oLog)
- {
- $oLog->Set('message', 'Error: '.$e->getMessage());
- }
- }
- if ($oLog)
- {
- $oLog->DBUpdate();
- }
- }
- protected function _DoExecute($oTrigger, $aContextArgs, &$oLog)
- {
- $sPreviousUrlMaker = ApplicationContext::SetUrlMakerClass();
- try
- {
- $this->m_iRecipients = 0;
- $this->m_aMailErrors = array();
- $bRes = false; // until we do succeed in sending the email
-
- // Determine recicipients
- //
- $sTo = $this->FindRecipients('to', $aContextArgs);
- $sCC = $this->FindRecipients('cc', $aContextArgs);
- $sBCC = $this->FindRecipients('bcc', $aContextArgs);
-
- $sFrom = MetaModel::ApplyParams($this->Get('from'), $aContextArgs);
- $sReplyTo = MetaModel::ApplyParams($this->Get('reply_to'), $aContextArgs);
-
- $sSubject = MetaModel::ApplyParams($this->Get('subject'), $aContextArgs);
- $sBody = MetaModel::ApplyParams($this->Get('body'), $aContextArgs);
-
- $oObj = $aContextArgs['this->object()'];
- $sMessageId = sprintf('iTop_%s_%d_%f@%s.openitop.org', get_class($oObj), $oObj->GetKey(), microtime(true /* get as float*/), MetaModel::GetEnvironmentId());
- $sReference = '<'.$sMessageId.'>';
- }
- catch(Exception $e)
- {
- ApplicationContext::SetUrlMakerClass($sPreviousUrlMaker);
- throw $e;
- }
- ApplicationContext::SetUrlMakerClass($sPreviousUrlMaker);
-
- if (!is_null($oLog))
- {
- // Note: we have to secure this because those values are calculated
- // inside the try statement, and we would like to keep track of as
- // many data as we could while some variables may still be undefined
- if (isset($sTo)) $oLog->Set('to', $sTo);
- if (isset($sCC)) $oLog->Set('cc', $sCC);
- if (isset($sBCC)) $oLog->Set('bcc', $sBCC);
- if (isset($sFrom)) $oLog->Set('from', $sFrom);
- if (isset($sSubject)) $oLog->Set('subject', $sSubject);
- if (isset($sBody)) $oLog->Set('body', $sBody);
- }
- $oEmail = new EMail();
- if ($this->IsBeingTested())
- {
- $oEmail->SetSubject('TEST['.$sSubject.']');
- $sTestBody = $sBody;
- $sTestBody .= "<div style=\"border: dashed;\">\n";
- $sTestBody .= "<h1>Testing email notification ".$this->GetHyperlink()."</h1>\n";
- $sTestBody .= "<p>The email should be sent with the following properties\n";
- $sTestBody .= "<ul>\n";
- $sTestBody .= "<li>TO: $sTo</li>\n";
- $sTestBody .= "<li>CC: $sCC</li>\n";
- $sTestBody .= "<li>BCC: $sBCC</li>\n";
- $sTestBody .= "<li>From: $sFrom</li>\n";
- $sTestBody .= "<li>Reply-To: $sReplyTo</li>\n";
- $sTestBody .= "<li>References: $sReference</li>\n";
- $sTestBody .= "</ul>\n";
- $sTestBody .= "</p>\n";
- $sTestBody .= "</div>\n";
- $oEmail->SetBody($sTestBody);
- $oEmail->SetRecipientTO($this->Get('test_recipient'));
- $oEmail->SetRecipientFrom($this->Get('test_recipient'));
- $oEmail->SetReferences($sReference);
- $oEmail->SetMessageId($sMessageId);
- }
- else
- {
- $oEmail->SetSubject($sSubject);
- $oEmail->SetBody($sBody);
- $oEmail->SetRecipientTO($sTo);
- $oEmail->SetRecipientCC($sCC);
- $oEmail->SetRecipientBCC($sBCC);
- $oEmail->SetRecipientFrom($sFrom);
- $oEmail->SetRecipientReplyTo($sReplyTo);
- $oEmail->SetReferences($sReference);
- $oEmail->SetMessageId($sMessageId);
- }
- if (isset($aContextArgs['attachments']))
- {
- $aAttachmentReport = array();
- foreach($aContextArgs['attachments'] as $oDocument)
- {
- $oEmail->AddAttachment($oDocument->GetData(), $oDocument->GetFileName(), $oDocument->GetMimeType());
- $aAttachmentReport[] = array($oDocument->GetFileName(), $oDocument->GetMimeType(), strlen($oDocument->GetData()));
- }
- $oLog->Set('attachments', $aAttachmentReport);
- }
- if (empty($this->m_aMailErrors))
- {
- if ($this->m_iRecipients == 0)
- {
- return 'No recipient';
- }
- else
- {
- $iRes = $oEmail->Send($aErrors, false, $oLog); // allow asynchronous mode
- switch ($iRes)
- {
- case EMAIL_SEND_OK:
- return "Sent";
-
- case EMAIL_SEND_PENDING:
- return "Pending";
-
- case EMAIL_SEND_ERROR:
- return "Errors: ".implode(', ', $aErrors);
- }
- }
- }
- else
- {
- if (is_array($this->m_aMailErrors) && count($this->m_aMailErrors) > 0)
- {
- $sError = implode(', ', $this->m_aMailErrors);
- }
- else
- {
- $sError = 'Unknown reason';
- }
- return 'Notification was not sent: '.$sError;
- }
- }
- }
- ?>
|