Liveagent - Version 1.0.15

Version Notes

changes:
- removes magento compilator errors

Download this release

Release Info

Developer Juraj Simon
Extension Liveagent
Version 1.0.15
Comparing to
See all releases


Version 1.0.15

Files changed (54) hide show
  1. app/code/local/Qualityunit/.buildpath +10 -0
  2. app/code/local/Qualityunit/.project +28 -0
  3. app/code/local/Qualityunit/.settings/.jsdtscope +11 -0
  4. app/code/local/Qualityunit/.settings/org.eclipse.php.core.prefs +5 -0
  5. app/code/local/Qualityunit/.settings/org.eclipse.wst.jsdt.ui.superType.container +1 -0
  6. app/code/local/Qualityunit/.settings/org.eclipse.wst.jsdt.ui.superType.name +1 -0
  7. app/code/local/Qualityunit/Liveagent/Block/Account.php +95 -0
  8. app/code/local/Qualityunit/Liveagent/Block/Accountbutton.php +34 -0
  9. app/code/local/Qualityunit/Liveagent/Block/Adminhtml/Config/Account.php +117 -0
  10. app/code/local/Qualityunit/Liveagent/Block/Adminhtml/Liveagent.php +12 -0
  11. app/code/local/Qualityunit/Liveagent/Block/Adminhtml/Liveagent/Grid.php +117 -0
  12. app/code/local/Qualityunit/Liveagent/Block/Base.php +50 -0
  13. app/code/local/Qualityunit/Liveagent/Block/ButtonsGrid.php +116 -0
  14. app/code/local/Qualityunit/Liveagent/Block/Connectioninfo.php +44 -0
  15. app/code/local/Qualityunit/Liveagent/Block/CreateFreeAccountWidget.php +18 -0
  16. app/code/local/Qualityunit/Liveagent/Block/CustomHeader.php +28 -0
  17. app/code/local/Qualityunit/Liveagent/Block/CustomHeader2.php +213 -0
  18. app/code/local/Qualityunit/Liveagent/Block/Dialog.php +5 -0
  19. app/code/local/Qualityunit/Liveagent/Block/ErrorMessageWidget.php +30 -0
  20. app/code/local/Qualityunit/Liveagent/Block/FloatButtonWidget.php +14 -0
  21. app/code/local/Qualityunit/Liveagent/Block/Liveagent.php +16 -0
  22. app/code/local/Qualityunit/Liveagent/Block/Signup.php +124 -0
  23. app/code/local/Qualityunit/Liveagent/Block/Waiting.php +80 -0
  24. app/code/local/Qualityunit/Liveagent/Exception/ConnectProblem.php +12 -0
  25. app/code/local/Qualityunit/Liveagent/Exception/SignupFailed.php +12 -0
  26. app/code/local/Qualityunit/Liveagent/Helper/Auth.php +79 -0
  27. app/code/local/Qualityunit/Liveagent/Helper/Base.php +52 -0
  28. app/code/local/Qualityunit/Liveagent/Helper/Buttons.php +143 -0
  29. app/code/local/Qualityunit/Liveagent/Helper/Data.php +6 -0
  30. app/code/local/Qualityunit/Liveagent/Helper/PhpApi.php +4310 -0
  31. app/code/local/Qualityunit/Liveagent/Helper/Settings.php +187 -0
  32. app/code/local/Qualityunit/Liveagent/Helper/Signup.php +26 -0
  33. app/code/local/Qualityunit/Liveagent/Model/Buttons.php +10 -0
  34. app/code/local/Qualityunit/Liveagent/Model/Liveagent.php +10 -0
  35. app/code/local/Qualityunit/Liveagent/Model/Mysql4/Buttons.php +9 -0
  36. app/code/local/Qualityunit/Liveagent/Model/Mysql4/Buttons/Collection.php +10 -0
  37. app/code/local/Qualityunit/Liveagent/Model/Mysql4/Liveagent.php +10 -0
  38. app/code/local/Qualityunit/Liveagent/Model/Mysql4/Liveagent/Collection.php +10 -0
  39. app/code/local/Qualityunit/Liveagent/Model/Mysql4/Settings.php +10 -0
  40. app/code/local/Qualityunit/Liveagent/Model/Mysql4/Settings/Collection.php +10 -0
  41. app/code/local/Qualityunit/Liveagent/Model/Settings.php +10 -0
  42. app/code/local/Qualityunit/Liveagent/Model/Status.php +15 -0
  43. app/code/local/Qualityunit/Liveagent/controllers/Adminhtml/ButtonsController.php +84 -0
  44. app/code/local/Qualityunit/Liveagent/controllers/Adminhtml/LiveagentController.php +242 -0
  45. app/code/local/Qualityunit/Liveagent/controllers/IndexController.php +9 -0
  46. app/code/local/Qualityunit/Liveagent/etc/config.xml +153 -0
  47. app/code/local/Qualityunit/Liveagent/sql/liveagent_setup/mysql4-install-0.1.0.php +59 -0
  48. app/design/adminhtml/default/default/layout/liveagent.xml +8 -0
  49. app/design/adminhtml/default/default/template/liveagent/dialog.phtml +1 -0
  50. app/design/frontend/base/default/layout/liveagent.xml +11 -0
  51. app/design/frontend/base/default/template/liveagent/button.phtml +10 -0
  52. app/design/frontend/base/default/template/liveagent/header.phtml +5 -0
  53. app/etc/modules/Qualityunit_Liveagent.xml +9 -0
  54. package.xml +20 -0
app/code/local/Qualityunit/.buildpath ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <buildpath>
3
+ <buildpathentry kind="con" path="org.eclipse.php.core.LANGUAGE"/>
4
+ <buildpathentry kind="src" path="Liveagent/Model"/>
5
+ <buildpathentry kind="src" path="Liveagent/Block"/>
6
+ <buildpathentry kind="src" path="Liveagent/Helper"/>
7
+ <buildpathentry kind="src" path="Liveagent/controllers"/>
8
+ <buildpathentry kind="src" path="Liveagent/sql/liveagent_setup"/>
9
+ <buildpathentry combineaccessrules="false" kind="prj" path="/magento"/>
10
+ </buildpath>
app/code/local/Qualityunit/.project ADDED
@@ -0,0 +1,28 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <projectDescription>
3
+ <name>la_code</name>
4
+ <comment></comment>
5
+ <projects>
6
+ </projects>
7
+ <buildSpec>
8
+ <buildCommand>
9
+ <name>org.eclipse.wst.jsdt.core.javascriptValidator</name>
10
+ <arguments>
11
+ </arguments>
12
+ </buildCommand>
13
+ <buildCommand>
14
+ <name>org.eclipse.wst.validation.validationbuilder</name>
15
+ <arguments>
16
+ </arguments>
17
+ </buildCommand>
18
+ <buildCommand>
19
+ <name>org.eclipse.dltk.core.scriptbuilder</name>
20
+ <arguments>
21
+ </arguments>
22
+ </buildCommand>
23
+ </buildSpec>
24
+ <natures>
25
+ <nature>org.eclipse.php.core.PHPNature</nature>
26
+ <nature>org.eclipse.wst.jsdt.core.jsNature</nature>
27
+ </natures>
28
+ </projectDescription>
app/code/local/Qualityunit/.settings/.jsdtscope ADDED
@@ -0,0 +1,11 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <classpath>
3
+ <classpathentry kind="con" path="org.eclipse.wst.jsdt.launching.JRE_CONTAINER"/>
4
+ <classpathentry kind="con" path="org.eclipse.wst.jsdt.launching.WebProject">
5
+ <attributes>
6
+ <attribute name="hide" value="true"/>
7
+ </attributes>
8
+ </classpathentry>
9
+ <classpathentry kind="con" path="org.eclipse.wst.jsdt.launching.baseBrowserLibrary"/>
10
+ <classpathentry kind="output" path=""/>
11
+ </classpath>
app/code/local/Qualityunit/.settings/org.eclipse.php.core.prefs ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ #Fri Sep 16 12:31:44 CEST 2011
2
+ eclipse.preferences.version=1
3
+ include_path=0;/la_code\u00052;/magento
4
+ phpVersion=php5.3
5
+ use_asp_tags_as_php=false
app/code/local/Qualityunit/.settings/org.eclipse.wst.jsdt.ui.superType.container ADDED
@@ -0,0 +1 @@
 
1
+ org.eclipse.wst.jsdt.launching.baseBrowserLibrary
app/code/local/Qualityunit/.settings/org.eclipse.wst.jsdt.ui.superType.name ADDED
@@ -0,0 +1 @@
 
1
+ Window
app/code/local/Qualityunit/Liveagent/Block/Account.php ADDED
@@ -0,0 +1,95 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_Account extends Qualityunit_Liveagent_Block_Base {
4
+
5
+ const SAVE_ACCOUNT_SETTINGS_ACTION_FLAG = 'sas';
6
+
7
+ public function _prepareLayout() {
8
+ parent::_prepareLayout();
9
+ $this->assignVariable('dialogCaption', Mage::helper('adminhtml')->__('Account options'));
10
+ $this->assignVariable('submitCaption', Mage::helper('adminhtml')->__('Save Account Settings'));
11
+ $this->assignVariable('formKey', Mage::getSingleton('core/session')->getFormKey());
12
+ $this->assignVariable('saveUrlAction', $this->getUrl('*/*/post'));
13
+ $this->assignVariable(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME, $this->getInputBox(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME, ''));
14
+ $this->assignVariable(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME, $this->getInputBox(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME, ''));
15
+ $this->assignVariable(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_PASSWORD_SETTING_NAME, $this->getPasswordBox(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_PASSWORD_SETTING_NAME, ''));
16
+ $this->assignVariable('saveActionSettingsFlag', self::SAVE_ACCOUNT_SETTINGS_ACTION_FLAG);
17
+ }
18
+
19
+ protected function getPasswordBox($name, $value) {
20
+ $settings = new Qualityunit_Liveagent_Helper_Settings();
21
+ return parent::getPasswordBox($name, $settings->getOption($name));
22
+ }
23
+
24
+ protected function getInputBox($name, $value) {
25
+ $settings = new Qualityunit_Liveagent_Helper_Settings();
26
+ return parent::getInputBox($name, $settings->getOption($name));
27
+ }
28
+
29
+ protected function getTemplateString() {
30
+ return '
31
+ <div class="wrap">
32
+ <div class="nlStepBox">
33
+ <div class="nlStepHeader">
34
+ <div class="nlStepHeaderLeft">
35
+ <span class="nlStepName">
36
+ Account options
37
+ </span>
38
+ </div>
39
+ <div class="clear"></div>
40
+ </div>
41
+ <form id="edit_form" name="edit_form" action="{saveUrlAction}" method="post">
42
+ <input name="form_key" type="hidden" value="{formKey}" />
43
+ <input name="action" type="hidden" value="{saveActionSettingsFlag}"/>
44
+ <div class="nlStepBoxContent">
45
+ <div class="nlStepBoxLeft">
46
+ <div class="nlFormField">
47
+ <div class="nlFormFieldLabel">
48
+ Account Url
49
+ </div>
50
+ <div class="nlFormFieldInput">
51
+ {la-url}
52
+ </div>
53
+ <div class="nlFormFieldDesc">
54
+ include \'http://\' or \'https://\'
55
+ </div>
56
+ </div>
57
+ <div class="nlFormField">
58
+ <div class="nlFormFieldLabel">
59
+ Email
60
+ </div>
61
+ <div class="nlFormFieldInput">
62
+ {la-owner-email}
63
+ </div>
64
+ </div>
65
+ <div class="nlFormField">
66
+ <div class="nlFormFieldLabel">
67
+ Password
68
+ </div>
69
+ <div class="nlFormFieldInput">
70
+ {la-owner-password}
71
+ </div>
72
+ </div>
73
+ <input type="hidden" value="submit">
74
+ <input class="nlBigButton" value="{submitCaption}" type="submit" style="border-width:0px;width:300px;"/>
75
+ </div>
76
+ <div class="nlBoxWithScreenshots"></div>
77
+ <div class="nlContactUs">
78
+ <div class="nlDarkOrangeText">
79
+ Do you need help with this plugin?
80
+ </div>
81
+ <div class="nlDarkOrangeText">
82
+ Feel free to
83
+ <a href="http://support.qualityunit.com/la/chat/index.php?bid=858b57ea" target="_blank">
84
+ contact us
85
+ </a>
86
+ </div>
87
+ </div>
88
+ <div class="clear"></div>
89
+ </div>
90
+ </form>
91
+ </div>
92
+ </div>
93
+ ';
94
+ }
95
+ }
app/code/local/Qualityunit/Liveagent/Block/Accountbutton.php ADDED
@@ -0,0 +1,34 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_Accountbutton extends Qualityunit_Liveagent_Block_Base {
4
+ /**
5
+ * @var Qualityunit_Liveagent_Helper_Settings
6
+ */
7
+ private $settings;
8
+
9
+ protected function _prepareLayout() {
10
+ parent::_prepareLayout();
11
+ $this->assignVariable('href', $this->getHref());
12
+ $this->assignVariable('caption', Mage::helper('adminhtml')->__('Login to Admin panel'));
13
+ }
14
+
15
+ public function setSettings(Qualityunit_Liveagent_Helper_Settings $settings) {
16
+ $this->settings = $settings;
17
+ }
18
+
19
+ private function getHref() {
20
+ $authToken = $this->settings->getOwnerAuthToken();
21
+ if ($authToken == '') {
22
+ $sessionId = $this->settings->getOwnerSessionId();
23
+ if ($sessionId != '') {
24
+ return $this->settings->getLiveAgentUrl() . '/agent?S='.$this->settings->getOwnerSessionId();
25
+ }
26
+ return $this->settings->getLiveAgentUrl() . '/agent';
27
+ }
28
+ return $this->settings->getLiveAgentUrl() . '/agent?AuthToken='.$authToken;
29
+ }
30
+
31
+ protected function getTemplateString() {
32
+ return '<a href="{href}" target="_blank" class="nlBigButton" style="width:300px;">{caption}</a>';
33
+ }
34
+ }
app/code/local/Qualityunit/Liveagent/Block/Adminhtml/Config/Account.php ADDED
@@ -0,0 +1,117 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_Adminhtml_Liveagent_Grid extends Mage_Adminhtml_Block_Widget_Grid
4
+ {
5
+
6
+ public function __construct()
7
+ {
8
+ parent::__construct();
9
+ $this->setId('liveagentGrid');
10
+ $this->setDefaultSort('liveagent_id');
11
+ $this->setDefaultDir('ASC');
12
+ $this->setSaveParametersInSession(true);
13
+ }
14
+
15
+ protected function _prepareCollection()
16
+ {
17
+ $collection = Mage::getModel('liveagent/liveagent')->getCollection();
18
+ $this->setCollection($collection);
19
+ return parent::_prepareCollection();
20
+ }
21
+
22
+ protected function _prepareColumns()
23
+ {
24
+ $this->addColumn('liveagent_id', array(
25
+ 'header' => Mage::helper('liveagent')->__('ID'),
26
+ 'align' =>'right',
27
+ 'width' => '50px',
28
+ 'index' => 'liveagent_id',
29
+ ));
30
+
31
+ $this->addColumn('title', array(
32
+ 'header' => Mage::helper('liveagent')->__('Title'),
33
+ 'align' =>'left',
34
+ 'index' => 'title',
35
+ ));
36
+
37
+ /*
38
+ $this->addColumn('content', array(
39
+ 'header' => Mage::helper('liveagent')->__('Item Content'),
40
+ 'width' => '150px',
41
+ 'index' => 'content',
42
+ ));
43
+ */
44
+
45
+ $this->addColumn('status', array(
46
+ 'header' => Mage::helper('liveagent')->__('Status'),
47
+ 'align' => 'left',
48
+ 'width' => '80px',
49
+ 'index' => 'status',
50
+ 'type' => 'options',
51
+ 'options' => array(
52
+ 1 => 'Enabled',
53
+ 2 => 'Disabled',
54
+ ),
55
+ ));
56
+
57
+ $this->addColumn('action',
58
+ array(
59
+ 'header' => Mage::helper('liveagent')->__('Action'),
60
+ 'width' => '100',
61
+ 'type' => 'action',
62
+ 'getter' => 'getId',
63
+ 'actions' => array(
64
+ array(
65
+ 'caption' => Mage::helper('liveagent')->__('Edit'),
66
+ 'url' => array('base'=> '*/*/edit'),
67
+ 'field' => 'id'
68
+ )
69
+ ),
70
+ 'filter' => false,
71
+ 'sortable' => false,
72
+ 'index' => 'stores',
73
+ 'is_system' => true,
74
+ ));
75
+
76
+ $this->addExportType('*/*/exportCsv', Mage::helper('liveagent')->__('CSV'));
77
+ $this->addExportType('*/*/exportXml', Mage::helper('liveagent')->__('XML'));
78
+
79
+ return parent::_prepareColumns();
80
+ }
81
+
82
+ protected function _prepareMassaction()
83
+ {
84
+ $this->setMassactionIdField('liveagent_id');
85
+ $this->getMassactionBlock()->setFormFieldName('liveagent');
86
+
87
+ $this->getMassactionBlock()->addItem('delete', array(
88
+ 'label' => Mage::helper('liveagent')->__('Delete'),
89
+ 'url' => $this->getUrl('*/*/massDelete'),
90
+ 'confirm' => Mage::helper('liveagent')->__('Are you sure?')
91
+ ));
92
+
93
+ $statuses = Mage::getSingleton('liveagent/status')->getOptionArray();
94
+
95
+ array_unshift($statuses, array('label'=>'', 'value'=>''));
96
+ $this->getMassactionBlock()->addItem('status', array(
97
+ 'label'=> Mage::helper('liveagent')->__('Change status'),
98
+ 'url' => $this->getUrl('*/*/massStatus', array('_current'=>true)),
99
+ 'additional' => array(
100
+ 'visibility' => array(
101
+ 'name' => 'status',
102
+ 'type' => 'select',
103
+ 'class' => 'required-entry',
104
+ 'label' => Mage::helper('liveagent')->__('Status'),
105
+ 'values' => $statuses
106
+ )
107
+ )
108
+ ));
109
+ return $this;
110
+ }
111
+
112
+ public function getRowUrl($row)
113
+ {
114
+ return $this->getUrl('*/*/edit', array('id' => $row->getId()));
115
+ }
116
+
117
+ }
app/code/local/Qualityunit/Liveagent/Block/Adminhtml/Liveagent.php ADDED
@@ -0,0 +1,12 @@
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Qualityunit_Liveagent_Block_Adminhtml_Liveagent extends Mage_Adminhtml_Block_Widget_Grid_Container
3
+ {
4
+ public function __construct()
5
+ {
6
+ $this->_controller = 'adminhtml_liveagent';
7
+ $this->_blockGroup = 'liveagent';
8
+ $this->_headerText = Mage::helper('liveagent')->__('Item Manager');
9
+ $this->_addButtonLabel = Mage::helper('liveagent')->__('Add Item');
10
+ parent::__construct();
11
+ }
12
+ }
app/code/local/Qualityunit/Liveagent/Block/Adminhtml/Liveagent/Grid.php ADDED
@@ -0,0 +1,117 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_Adminhtml_Liveagent_Grid extends Mage_Adminhtml_Block_Widget_Grid
4
+ {
5
+
6
+ public function __construct()
7
+ {
8
+ parent::__construct();
9
+ $this->setId('liveagentGrid');
10
+ $this->setDefaultSort('liveagent_id');
11
+ $this->setDefaultDir('ASC');
12
+ $this->setSaveParametersInSession(true);
13
+ }
14
+
15
+ protected function _prepareCollection()
16
+ {
17
+ $collection = Mage::getModel('liveagent/liveagent')->getCollection();
18
+ $this->setCollection($collection);
19
+ return parent::_prepareCollection();
20
+ }
21
+
22
+ protected function _prepareColumns()
23
+ {
24
+ $this->addColumn('liveagent_id', array(
25
+ 'header' => Mage::helper('liveagent')->__('ID'),
26
+ 'align' =>'right',
27
+ 'width' => '50px',
28
+ 'index' => 'liveagent_id',
29
+ ));
30
+
31
+ $this->addColumn('title', array(
32
+ 'header' => Mage::helper('liveagent')->__('Title'),
33
+ 'align' =>'left',
34
+ 'index' => 'title',
35
+ ));
36
+
37
+ /*
38
+ $this->addColumn('content', array(
39
+ 'header' => Mage::helper('liveagent')->__('Item Content'),
40
+ 'width' => '150px',
41
+ 'index' => 'content',
42
+ ));
43
+ */
44
+
45
+ $this->addColumn('status', array(
46
+ 'header' => Mage::helper('liveagent')->__('Status'),
47
+ 'align' => 'left',
48
+ 'width' => '80px',
49
+ 'index' => 'status',
50
+ 'type' => 'options',
51
+ 'options' => array(
52
+ 1 => 'Enabled',
53
+ 2 => 'Disabled',
54
+ ),
55
+ ));
56
+
57
+ $this->addColumn('action',
58
+ array(
59
+ 'header' => Mage::helper('liveagent')->__('Action'),
60
+ 'width' => '100',
61
+ 'type' => 'action',
62
+ 'getter' => 'getId',
63
+ 'actions' => array(
64
+ array(
65
+ 'caption' => Mage::helper('liveagent')->__('Edit'),
66
+ 'url' => array('base'=> '*/*/edit'),
67
+ 'field' => 'id'
68
+ )
69
+ ),
70
+ 'filter' => false,
71
+ 'sortable' => false,
72
+ 'index' => 'stores',
73
+ 'is_system' => true,
74
+ ));
75
+
76
+ $this->addExportType('*/*/exportCsv', Mage::helper('liveagent')->__('CSV'));
77
+ $this->addExportType('*/*/exportXml', Mage::helper('liveagent')->__('XML'));
78
+
79
+ return parent::_prepareColumns();
80
+ }
81
+
82
+ protected function _prepareMassaction()
83
+ {
84
+ $this->setMassactionIdField('liveagent_id');
85
+ $this->getMassactionBlock()->setFormFieldName('liveagent');
86
+
87
+ $this->getMassactionBlock()->addItem('delete', array(
88
+ 'label' => Mage::helper('liveagent')->__('Delete'),
89
+ 'url' => $this->getUrl('*/*/massDelete'),
90
+ 'confirm' => Mage::helper('liveagent')->__('Are you sure?')
91
+ ));
92
+
93
+ $statuses = Mage::getSingleton('liveagent/status')->getOptionArray();
94
+
95
+ array_unshift($statuses, array('label'=>'', 'value'=>''));
96
+ $this->getMassactionBlock()->addItem('status', array(
97
+ 'label'=> Mage::helper('liveagent')->__('Change status'),
98
+ 'url' => $this->getUrl('*/*/massStatus', array('_current'=>true)),
99
+ 'additional' => array(
100
+ 'visibility' => array(
101
+ 'name' => 'status',
102
+ 'type' => 'select',
103
+ 'class' => 'required-entry',
104
+ 'label' => Mage::helper('liveagent')->__('Status'),
105
+ 'values' => $statuses
106
+ )
107
+ )
108
+ ));
109
+ return $this;
110
+ }
111
+
112
+ public function getRowUrl($row)
113
+ {
114
+ return $this->getUrl('*/*/edit', array('id' => $row->getId()));
115
+ }
116
+
117
+ }
app/code/local/Qualityunit/Liveagent/Block/Base.php ADDED
@@ -0,0 +1,50 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_Base extends Mage_Core_Block_Template {
4
+
5
+ protected $variables = array();
6
+
7
+ protected function curPageURL() {
8
+ $pageURL = 'http';
9
+ if (isset($_SERVER["HTTPS"]) && $_SERVER["HTTPS"] == "on") {$pageURL .= "s";}
10
+ $pageURL .= "://";
11
+ if ($_SERVER["SERVER_PORT"] != "80") {
12
+ $pageURL .= $_SERVER["SERVER_NAME"].":".$_SERVER["SERVER_PORT"].$_SERVER["REQUEST_URI"];
13
+ } else {
14
+ $pageURL .= $_SERVER["SERVER_NAME"].$_SERVER["REQUEST_URI"];
15
+ }
16
+
17
+ $pageURL = preg_replace("/\?.*$/", "", $pageURL);
18
+
19
+ return $pageURL;
20
+ }
21
+
22
+ protected function getPasswordBox($name, $value) {
23
+ return '<input type="password" id="'.$name.'" name="'.$name.'" value="'.$value.'" class="nlInput" />';
24
+ }
25
+
26
+ protected function getInputBox($name, $value) {
27
+ return '<input type="text" id="'.$name.'" name="'.$name.'" value="'.$value.'" class="nlInput" />';
28
+ }
29
+
30
+ protected function _toHtml() {
31
+ $html = $this->getTemplateString();
32
+ foreach ($this->variables as $name => $value) {
33
+ $html = str_replace('{'.$name.'}', $value, $html);
34
+ }
35
+ return $html;
36
+ }
37
+
38
+ protected function _prepareLayout() {
39
+ parent::_prepareLayout();
40
+ //$this->getLayout()->getBlock('head')->addCss('css/liveagent/liveagent.css');
41
+ }
42
+
43
+ protected function assignVariable($name, $value) {
44
+ $this->variables[$name] = $value;
45
+ }
46
+
47
+ protected function getTemplateString() {
48
+ return '';
49
+ }
50
+ }
app/code/local/Qualityunit/Liveagent/Block/ButtonsGrid.php ADDED
@@ -0,0 +1,116 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_ButtonsGrid extends Mage_Adminhtml_Block_Widget_Grid
4
+ {
5
+
6
+ public function __construct()
7
+ {
8
+ parent::__construct();
9
+ $this->setId('liveagentGrid');
10
+ $this->setDefaultSort('buttonid');
11
+ $this->setDefaultDir('ASC');
12
+ $this->setSaveParametersInSession(true);
13
+ }
14
+
15
+ protected function _prepareCollection()
16
+ {
17
+ $collection = Mage::getModel('liveagent/buttons')->getCollection()->addFilter('contenttype', 'F');
18
+ $this->setCollection($collection);
19
+ return parent::_prepareCollection();
20
+ }
21
+
22
+ protected function _prepareColumns()
23
+ {
24
+ $this->addColumn('buttonid', array(
25
+ 'header' => Mage::helper('liveagent')->__('ID'),
26
+ 'align' =>'right',
27
+ 'width' => '50px',
28
+ 'index' => 'buttonid',
29
+ ));
30
+
31
+ $this->addColumn('name', array(
32
+ 'header' => Mage::helper('liveagent')->__('Name'),
33
+ 'align' =>'left',
34
+ 'index' => 'name',
35
+ ));
36
+
37
+ /*
38
+ $this->addColumn('content', array(
39
+ 'header' => Mage::helper('liveagent')->__('Item Content'),
40
+ 'width' => '150px',
41
+ 'index' => 'content',
42
+ ));
43
+ */
44
+
45
+ $this->addColumn('status', array(
46
+ 'header' => Mage::helper('liveagent')->__('Status'),
47
+ 'align' => 'left',
48
+ 'width' => '80px',
49
+ 'index' => 'onlinestatus',
50
+ 'type' => 'options',
51
+ 'options' => array(
52
+ 'Y' => 'Enabled',
53
+ 'N' => 'Disabled',
54
+ ),
55
+ ));
56
+
57
+ /*$this->addColumn('action',
58
+ array(
59
+ 'header' => Mage::helper('liveagent')->__('Action'),
60
+ 'width' => '100',
61
+ 'type' => 'action',
62
+ 'getter' => 'getId',
63
+ 'actions' => array(
64
+ array(
65
+ 'caption' => Mage::helper('liveagent')->__('Edit'),*/
66
+ //'url' => array('base'=> '*/*/edit'),
67
+ /*'field' => 'id'
68
+ )
69
+ ),
70
+ 'filter' => false,
71
+ 'sortable' => false,
72
+ 'index' => 'stores',
73
+ 'is_system' => true,
74
+ ));*/
75
+
76
+ //$this->addExportType('*/*/exportCsv', Mage::helper('liveagent')->__('CSV'));
77
+ //$this->addExportType('*/*/exportXml', Mage::helper('liveagent')->__('XML'));
78
+
79
+ return parent::_prepareColumns();
80
+ }
81
+
82
+ protected function _prepareMassaction()
83
+ {
84
+ $this->setMassactionIdField('liveagentbutton_id');
85
+ $this->getMassactionBlock()->setFormFieldName('buttonids');
86
+
87
+ // $this->getMassactionBlock()->addItem('delete', array(
88
+ // 'label' => Mage::helper('liveagent')->__('Delete'),
89
+ // 'url' => $this->getUrl('*/*/massDelete'),
90
+ // 'confirm' => Mage::helper('liveagent')->__('Are you sure?')
91
+ // ));
92
+
93
+ $statuses = Mage::getSingleton('liveagent/status')->getOptionArray();
94
+ array_unshift($statuses, array('label'=>'', 'value'=>''));
95
+ $this->getMassactionBlock()->addItem('status', array(
96
+ 'label'=> Mage::helper('liveagent')->__('Change status'),
97
+ 'url' => $this->getUrl('*/*/massStatus', array('_current'=>true)),
98
+ 'additional' => array(
99
+ 'visibility' => array(
100
+ 'name' => 'status',
101
+ 'type' => 'select',
102
+ 'class' => 'required-entry',
103
+ 'label' => Mage::helper('liveagent')->__('Status'),
104
+ 'values' => $statuses
105
+ )
106
+ )
107
+ ));
108
+ return $this;
109
+ }
110
+
111
+ // public function getRowUrl($row)
112
+ // {
113
+ // return $this->getUrl('*/*/edit', array('id' => $row->getId()));
114
+ // }
115
+
116
+ }
app/code/local/Qualityunit/Liveagent/Block/Connectioninfo.php ADDED
@@ -0,0 +1,44 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_Connectioninfo extends Qualityunit_Liveagent_Block_Base {
4
+ private $connectionValid;
5
+
6
+ public function setConnectionValid($valid) {
7
+ $this->connectionValid = $valid;
8
+ }
9
+
10
+ public function _prepareLayout() {
11
+ parent::_prepareLayout();
12
+ $this->assignVariable('connectionInfo', $this->getConnectionInfo());
13
+ }
14
+
15
+ /**
16
+ * @return string
17
+ */
18
+ private function getConnectionInfo() {
19
+ if ($this->connectionValid) {
20
+ return $this->getConnectionOk();
21
+ }
22
+ return $this->getConnectionError();
23
+ }
24
+
25
+ private function getConnectionOk() {
26
+ return '
27
+ <div class="nlHurrayMessage">
28
+ '.Mage::helper('adminhtml')->__('Your Magento installation is succesfully connected with Live Agent').'
29
+ </div>
30
+ ';
31
+ }
32
+
33
+ private function getConnectionError() {
34
+ return '
35
+ <div class="nlHurrayMessage">
36
+ '.Mage::helper('adminhtml')->__('Unable to connect, check your account settings').'
37
+ </div>
38
+ ';
39
+ }
40
+
41
+ protected function getTemplateString() {
42
+ return '{connectionInfo}';
43
+ }
44
+ }
app/code/local/Qualityunit/Liveagent/Block/CreateFreeAccountWidget.php ADDED
@@ -0,0 +1,18 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_CreateFreeAccountWidget extends Qualityunit_Liveagent_Block_Base {
4
+ public function _prepareLayout() {
5
+ parent::_prepareLayout();
6
+ $this->assignVariable('preMessage', Mage::helper('adminhtml')->__('No account selected. Enter your existing account credetials, or create new'));
7
+ $this->assignVariable('link', '<a href="'.$this->curPageURL() . '?action='.Qualityunit_Liveagent_Adminhtml_LiveagentController::ACTION_CREATE.'">' . Mage::helper('adminhtml')->__('free account') . '</a>');
8
+ $this->assignVariable('postMessage', Mage::helper('adminhtml')->__('(No credit card required)'));
9
+ }
10
+
11
+ protected function getTemplateString() {
12
+ return '
13
+ <div class="wrap">
14
+ {preMessage} {link}. {postMessage}
15
+ </div>
16
+ ';
17
+ }
18
+ }
app/code/local/Qualityunit/Liveagent/Block/CustomHeader.php ADDED
@@ -0,0 +1,28 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Qualityunit_Liveagent_Block_CustomHeader extends Mage_Page_Block_Html_Head {
3
+
4
+ private function includePhpApi() {
5
+ $path = dirname(__FILE__);
6
+ if (file_exists($path . '/../Helper/PhpApi.php')) {
7
+ require_once $path . '/../Helper/PhpApi.php';
8
+ return;
9
+ }
10
+ require_once ('Qualityunit_Liveagent_Helper_PhpApi.php');
11
+ }
12
+
13
+ protected function _construct() {
14
+ parent::_construct();
15
+ $settings = new Qualityunit_Liveagent_Helper_Settings();
16
+ if ((time() - $settings->getButtonsLastReloadTime()) > 120) {
17
+ Mage::log('Buttons outdated. Refreshing from application...', Zend_Log::DEBUG);
18
+ $buttons = new Qualityunit_Liveagent_Helper_Buttons();
19
+ try {
20
+ $this->includePhpApi();
21
+ $settings->login();
22
+ $buttons->refresh();
23
+ } catch (Exception $e) {
24
+ Mage::log('Unable to refresh buttons', Zend_Log::ERR);
25
+ }
26
+ }
27
+ }
28
+ }
app/code/local/Qualityunit/Liveagent/Block/CustomHeader2.php ADDED
@@ -0,0 +1,213 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_CustomHeader2 extends Mage_Adminhtml_Block_Page_Head {
4
+ private function includePhpApi() {
5
+ $path = dirname(__FILE__);
6
+ if (file_exists($path . '/../Helper/PhpApi.php')) {
7
+ require_once $path . '/../Helper/PhpApi.php';
8
+ return;
9
+ }
10
+ require_once ('Qualityunit_Liveagent_Helper_PhpApi.php');
11
+ }
12
+
13
+ protected function _construct() {
14
+ parent::_construct();
15
+ $settings = new Qualityunit_Liveagent_Helper_Settings();
16
+ if ((time() - $settings->getButtonsLastReloadTime()) > 120) {
17
+ Mage::log('Buttons outdated. Refreshing from application...', Zend_Log::DEBUG);
18
+ $buttons = new Qualityunit_Liveagent_Helper_Buttons();
19
+ try {
20
+ $this->includePhpApi();
21
+ $settings->login();
22
+ $buttons->refresh();
23
+ } catch (Exception $e) {
24
+ Mage::log('Unable to refresh buttons', Zend_Log::ERR);
25
+ }
26
+ }
27
+ }
28
+
29
+ protected function _toHtml() {
30
+ return parent::_toHtml() . '
31
+ <style type="text/css">
32
+ /* CLASSES from Juraj */
33
+
34
+ div.liveagentInfo {
35
+ background-color: #D0E4F5;
36
+ border-color: #52AAF2;
37
+ }
38
+
39
+ div.laSignup {
40
+ display: block;
41
+ }
42
+
43
+ .buttonTableRow {
44
+ display: inline;
45
+ clear: both;
46
+ float: left;
47
+ position: relative;
48
+ background: #ffb347;
49
+ margin-bottom:3px;
50
+ }
51
+
52
+ .laButtonPreview { /*position: absolute;*/
53
+ overflow:hidden;
54
+ }
55
+
56
+ .laButtonPreview * {
57
+ border:none;
58
+ }
59
+
60
+
61
+ .laButtonsTableCell {
62
+ position: relative;
63
+ float: left;
64
+ width: 300px;
65
+ height: 100px;
66
+ vertical-align: middle;
67
+ text-align: center;
68
+ }
69
+
70
+ .laButtonsTableHeader {
71
+ position: relative;
72
+ float: left;
73
+ overflow: auto;
74
+ width: 300px;
75
+ text-align: center;
76
+ }
77
+
78
+ .laButtonsTable {
79
+ position: relative;
80
+ clear: both;
81
+ }
82
+
83
+ .laButtonsTableSubmit {
84
+ clear: both;
85
+ float: left;
86
+ }
87
+
88
+ .laButtonsTableHeaderType,
89
+ .laButtonsTableCellType {
90
+ width:150px;
91
+ }
92
+
93
+ .laButtonsTableHeaderEnabled,
94
+ .laButtonsTableCellEnabled {
95
+ width:150px;
96
+ }
97
+
98
+ .laButtonsTableCellType,
99
+ .laButtonsTableCellEnabled {
100
+ line-height:100px;
101
+ position:relative;
102
+ }
103
+
104
+ .laCheckboxWrap {
105
+ position:absolute;
106
+ top:50%;
107
+ left:50%;
108
+ width:14px;
109
+ height:14px;
110
+ }
111
+ .laCheckboxWrap input {
112
+ position:relative;
113
+ top:-5px;
114
+ left:-5px;
115
+ margin:0;
116
+ }
117
+
118
+ .VisitsTableHeader {
119
+ font-weight: bold;
120
+ }
121
+
122
+ .VisitorCell{
123
+ }
124
+
125
+ .VisitorLocation {
126
+ float: left;
127
+ clear: both;
128
+ font-weight: bold;
129
+ }
130
+
131
+ .VisitorFirstTime, .VisitorLastTime, .VisitorName, .VisitorRefUrl, .VisitorUrl {
132
+ float: left;
133
+ clear: both;
134
+ }
135
+
136
+ /* NEW CLASSES */
137
+
138
+ .nlFormFieldDomain nlDomainSelection {float:left;width: 186px;}
139
+
140
+ .clear {clear:both;}
141
+ .nlFormField {margin:0 0 15px 0;}
142
+ .nlFormFieldLabel {font-size:13px; font-weight:bold;}
143
+ .nlFormFieldInput, .nlFormFieldInputDomain {background:#fff; border:1px solid #949494; border-color:#888888 #AAAAAA #DDDDDD; border-radius:7px 7px 7px 7px; border-style:solid; border-width:1px; height:35px;}
144
+ .nlInput, .nlFormFieldInput .password-focus, .nlFormFieldInput .password, .nlFormFieldInput .text, .nlFormFieldInput .text-focus {font-size:17px !important; border-radius:5px 5px 5px 5px; height:100%; margin:0; border:none; padding:0 6px; width:275px;}
145
+ .nlFormFieldInputDomain .text-focus, .nlFormFieldInputDomain .text {font-size:17px !important; border-radius:5px 5px 5px 5px; height:100%; margin:0; border:none; padding:0 6px; width:186px;}
146
+
147
+ .nlStepBox {margin:20px 0; width:962px; font-family:\'PT Sans\', arial, serif; position:relative;}
148
+ .nlStepHeader {background:#1a1a1a url(\'#\') repeat-x; border-radius:8px 8px 0 0; -moz-border-radius:8px 8px 0 0; -webkit-border-radius:8px 8px 0 0; padding:18px 20px; text-align:left; font-size:25px; font-weight:bold; color:#bebebe;}
149
+ .nlStepHeaderLeft {float:left;}
150
+ .nlStepNumber {font-weight:normal;}
151
+ .nlStepHeaderRight {float:right; text-align:right; font-weight:normal; line-height:18px;}
152
+ .nlStepBoxContent {min-height:320px; _height:320px; background:#d9d9d9; border-radius:0 0 8px 8px; -moz-border-radius:0 0 8px 8px; -webkit-border-radius:0 0 8px 8px; border:1px solid #fc8700; padding:15px 0; background: -webkit-gradient(linear, left top, left bottom, from(#fd8100), to(#fdab00)); background:-moz-linear-gradient(top,#fd8100,#fdab00); filter:progid:DXImageTransform.Microsoft.gradient(startColorstr=\'#fd8100\', endColorstr=\'#fdab00\');}
153
+ .nlStepBoxLeft {width:300px; float:left; margin:0 20px;}
154
+ .nlBoxWithScreenshots {position:absolute; top:50px; left:350px; width:590px; height:395px; background:url() no-repeat top left;}
155
+ .nlDomainSelection {width:200px; float:left;}
156
+ .nlDomainSelection .nlInput {width:186px;}
157
+ .nlDomainName {width:80px; float:left; font-size:13px; margin:10px 0 0 10px;}
158
+ .nlTermsAndConditions {font-size:11px; font-family:"Trebuchet MS",Arial,Verdana; margin:0 0 15px;}
159
+ .nlTermsAndConditions a {color:#26813f;}
160
+ .nlTermsAndConditions a:hover {text-decoration:none;}
161
+ .nlContactUs {position:absolute; right:25px; bottom:20px; text-align:right;}
162
+ .nlOwnButtons {position:absolute; left:25px; bottom:20px; text-align:left;}
163
+ .nlGridHolder {padding:20px;}
164
+
165
+ .nlAboveStepBox {font-family:\'PT Sans\', arial, serif;}
166
+ .nlAboveStepBox .nlStepBoxLeft {margin:0;}
167
+
168
+ .nlBigButton {text-decoration:none; cursor:pointer; text-align:center; font-size:23px; line-height:50px; color:#295700; padding:0; clear:both; margin:10px 0 5px; display:block; box-shadow:#222 0 1px 2px; -moz-box-shadow:#222 0 1px 2px; webkit-box-shadow:#222 0 1px 2px; font-weight:bold; text-shadow:#cff400 0px 1px 1px; color:#295700; display:block; background:#5ac600; -moz-border-radius:5px; -webkit-border-radius:5px; border-radius:5px 5px 5px 5px; text-decoration:none; text-align:center; background: -webkit-gradient(linear, left top, left bottom, from(#c2e800), to(#5ec400)); background:-moz-linear-gradient(top,#c2e800,#5ec400); filter: progid:DXImageTransform.Microsoft.gradient(startColorstr=\'#c2e800\', endColorstr=\'#5ec400\');}
169
+ .nlBigButton:hover {background:#7fe710; background: -webkit-gradient(linear, left top, left bottom, from(#d1f200), to(#8bdd00)); background:-moz-linear-gradient(top,#d1f200,#8bdd00); filter: progid:DXImageTransform.Microsoft.gradient(startColorstr=\'#d1f200\', endColorstr=\'#8bdd00\');}
170
+ .nlBigButton.nlSmallText {font-size:14px;}
171
+
172
+ .nlDarkOrangeText {color:#7d4000; font-size:14px;}
173
+ .nlDarkOrangeText.nlSmallText {font-size:12px;}
174
+ .nlDarkOrangeText a {color:#7d4000;}
175
+ .nlDarkOrangeText a:hover {color:#7d4000; text-decoration:none;}
176
+
177
+ .nlDarkGrayText {color:#333; font-size:14px; margin:0 0 10px;}
178
+ .nlDarkGrayText a {color:#333;}
179
+ .nlDarkGrayText a:hover {color:#333; text-decoration:none;}
180
+
181
+ .nlLightText {color:#bebebe; font-size:14px;}
182
+ .nlLightText a {color:#bebebe;}
183
+ .nlLightText a:hover {color:#bebebe; text-decoration:none;}
184
+
185
+ .nlHurrayMessage {font-size:17px; font-weight:bold; text-align:left; margin:0 0 10px;}
186
+
187
+ .nlLoader {height:50px; background:url() no-repeat center center; margin:20px 0;}
188
+ .nlLoader span {display:none;}
189
+ .buttonsSave {
190
+ width:150px;
191
+ clear:both;
192
+ float: left;
193
+ }
194
+ .buttonsSettings .nlStepHeader {
195
+ width: 902px;
196
+ }
197
+ .buttonsSettings .nlStepBoxContent {
198
+ width:900px;
199
+ float:left;
200
+ clear:both;
201
+ }
202
+ .nlFormFieldDesc {
203
+ font-size: 11px;
204
+ }
205
+
206
+ .nlFormFieldInputDomain {
207
+ float:left;
208
+ }
209
+
210
+ </style>
211
+ ';
212
+ }
213
+ }
app/code/local/Qualityunit/Liveagent/Block/Dialog.php ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_Dialog extends Qualityunit_Liveagent_Block_Base {
4
+
5
+ }
app/code/local/Qualityunit/Liveagent/Block/ErrorMessageWidget.php ADDED
@@ -0,0 +1,30 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_ErrorMessageWidget extends Qualityunit_Liveagent_Block_Base {
4
+
5
+ private $messages = array();
6
+
7
+ public function addMessage($message) {
8
+ $this->messages[] = $message;
9
+ }
10
+
11
+ protected function _toHtml() {
12
+ $messagesString = '';
13
+ foreach ($this->messages as $message) {
14
+ $messagesString .= "
15
+ <div class=\"laMessage\">".$message."</div>\n
16
+ ";
17
+ }
18
+ $this->assignVariable('message', $messagesString);
19
+ return parent::_toHtml();
20
+
21
+ }
22
+
23
+ protected function getTemplateString() {
24
+ return '
25
+ <div class="wrap">
26
+ {message}
27
+ </div>
28
+ ';
29
+ }
30
+ }
app/code/local/Qualityunit/Liveagent/Block/FloatButtonWidget.php ADDED
@@ -0,0 +1,14 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_FloatButtonWidget extends Qualityunit_Liveagent_Block_Base {
4
+
5
+ private $buttonCode = '';
6
+
7
+ public function setCode($code) {
8
+ $this->buttonCode = $code;
9
+ }
10
+
11
+ protected function getTemplateString() {
12
+ return $this->buttonCode;
13
+ }
14
+ }
app/code/local/Qualityunit/Liveagent/Block/Liveagent.php ADDED
@@ -0,0 +1,16 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Qualityunit_Liveagent_Block_Liveagent extends Mage_Core_Block_Template
3
+ {
4
+ public function _prepareLayout()
5
+ {
6
+ return parent::_prepareLayout();
7
+ }
8
+
9
+ public function getLiveagent()
10
+ {
11
+ if (!$this->hasData('liveagent')) {
12
+ $this->setData('liveagent', Mage::registry('liveagent'));
13
+ }
14
+ return $this->getData('liveagent');
15
+ }
16
+ }
app/code/local/Qualityunit/Liveagent/Block/Signup.php ADDED
@@ -0,0 +1,124 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Block_Signup extends Qualityunit_Liveagent_Block_Base {
4
+
5
+ const FULL_NAME_FIELD = 'la-full-name';
6
+
7
+ public function _prepareLayout() {
8
+ parent::_prepareLayout();
9
+ $this->assignVariable('dialogCaption', Mage::helper('adminhtml')->__('Create your Free account'));
10
+ $this->assignVariable(self::FULL_NAME_FIELD, $this->getInputBox(self::FULL_NAME_FIELD, $this->getOwnerName()));
11
+ $this->assignVariable(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME, $this->getInputBox(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME, $this->getOwnerEmail()));
12
+ $this->assignVariable(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME, $this->getInputBox(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME, $this->getdomainOnly()));
13
+ $this->assignVariable('submitCaption', Mage::helper('adminhtml')->__('Create your account'));
14
+ $this->assignVariable('skipUrlAction', $this->curPageURL() . '?action=s');
15
+ $this->assignVariable('registerUrlAction', $this->getUrl('*/*/post'));
16
+ $this->assignVariable('formKey', Mage::getSingleton('core/session')->getFormKey());
17
+ }
18
+
19
+ private function getdomainOnly() {
20
+ $domain = preg_replace('/^(.*\.)?([^.]*\..*)$/', '$2', @$_SERVER['HTTP_HOST']);
21
+ if (trim($domain) == 'localhost') {
22
+ return strtolower(str_replace(' ', '', $this->getOwnerName())) . rand(100, 5000);
23
+ }
24
+ return $domain;
25
+ }
26
+
27
+ private function getOwnerName() {
28
+ try {
29
+ $user = Mage::getSingleton('admin/session')->getUser();
30
+ return $user->getFirstname() . ' ' . $user->getLastname();
31
+ } catch (Exception $e) {
32
+ Mage::log($e->getMessage(), Zend_Log::ERR);
33
+ return '';
34
+ }
35
+ }
36
+
37
+ private function getOwnerEmail() {
38
+ try {
39
+ $user = Mage::getSingleton('admin/session')->getUser();
40
+ return $user->getEmail();
41
+ } catch (Exception $e) {
42
+ Mage::log($e->getMessage(), Zend_Log::ERR);
43
+ return '';
44
+ }
45
+ }
46
+
47
+ protected function getTemplateString() {
48
+ return '
49
+ <div class="nlStepBox">
50
+ <div class="nlStepHeader">
51
+ <div class="nlStepHeaderLeft">
52
+ <span class="nlStepName">
53
+ {dialogCaption}
54
+ </span>
55
+ </div>
56
+ <div class="clear"></div>
57
+ </div>
58
+ <div class="nlStepBoxContent">
59
+ <div class="nlStepBoxLeft">
60
+ <form id="edit_form" name="edit_form" action="{registerUrlAction}" method="post">
61
+ <div class="nlFormField">
62
+ <div class="nlFormFieldLabel">
63
+ Full name
64
+ </div>
65
+ <div class="nlFormFieldInput">
66
+ {la-full-name}
67
+ </div>
68
+ </div>
69
+ <div class="nlFormField">
70
+ <div class="nlFormFieldLabel">
71
+ Email
72
+ </div>
73
+ <div class="nlFormFieldInput">
74
+ {la-owner-email}
75
+ </div>
76
+ </div>
77
+ <div class="nlFormField">
78
+ <div class="nlFormFieldLabel">
79
+ Domain selection
80
+ </div>
81
+ <div class="nlFormFieldInputDomain nlDomainSelection">
82
+ {la-url}
83
+ </div>
84
+ <div class="nlDomainName">
85
+ .ladesk.com
86
+ </div>
87
+ <div class="clear"></div>
88
+ </div>
89
+ <input name="form_key" type="hidden" value="{formKey}" />
90
+ <input name="action" type="hidden" value="r"/>
91
+ <input class="nlBigButton" value="{submitCaption}" type="submit" style="border-width:0px;width:300px;"/>
92
+ <div class="nlTermsAndConditions">
93
+ By creating an account I agree with
94
+ <a href="http://www.qualityunit.com/liveagent/pricing/hosted/terms-and-conditions" target="_blank">
95
+ Terms & Conditions
96
+ </a>
97
+ </div>
98
+ <div class="nlDarkOrangeText">
99
+ <a href="{skipUrlAction}" name="liveagent_link_onSignupCancel">
100
+ Skip this step, I have already account »
101
+ </a>
102
+ </div>
103
+ </form>
104
+ </div>
105
+ <div class="nlBoxWithScreenshots"></div>
106
+ <div class="nlContactUs">
107
+ <div class="nlDarkOrangeText">
108
+ Do you need help with this plugin?
109
+ </div>
110
+ <div class="nlDarkOrangeText">
111
+ Feel free to
112
+ <a href="http://support.qualityunit.com/la/chat/index.php?bid=858b57ea" target="_blank">
113
+ contact us
114
+ </a>
115
+ </div>
116
+ </div>
117
+ <div class="clear"></div>
118
+ </div>
119
+ </div>
120
+ </div>
121
+ <div class="clear"></div>
122
+ ';
123
+ }
124
+ }
app/code/local/Qualityunit/Liveagent/Block/Waiting.php ADDED
@@ -0,0 +1,80 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Qualityunit_Liveagent_Block_Waiting extends Qualityunit_Liveagent_Block_Base {
3
+ protected function _prepareLayout() {
4
+ parent::_prepareLayout();
5
+ $this->assignVariable('dialogCaption', Mage::helper('adminhtml')->__('Account Installation'));
6
+ $this->assignVariable('completeUrlAction', $this->getUrl('*/*/post'));
7
+ $this->assignVariable('formKey', Mage::getSingleton('core/session')->getFormKey());
8
+ }
9
+
10
+ protected function getTemplateString() {
11
+ return '
12
+ <div class="wrap">
13
+ <div class="nlStepBox">
14
+ <div class="nlStepHeader">
15
+ <div class="nlStepHeaderLeft">
16
+ <span class="nlStepName">
17
+ {dialogCaption}
18
+ </span>
19
+ </div>
20
+ <div class="clear"></div>
21
+ </div>
22
+ <div class="nlStepBoxContent">
23
+ <div class="nlStepBoxLeft">
24
+ <div class="nlHurrayMessage">
25
+ Thank you!
26
+ </div>
27
+
28
+ <div class="nlDarkOrangeText">
29
+ Your account will be online within next few minutes.
30
+ <br/><br/>
31
+ </div>
32
+ <div class="nlDarkOrangeText">
33
+ Please wait, once it will be online ....
34
+ <br/><br/>
35
+ </div>
36
+ <div class="nlDarkOrangeText">
37
+ You should recieve confirmation email with your account credentials. <br/><br/>
38
+ Note: Sometimes account creation process may take a <a href=\'http://support.qualityunit.com/knowledgebase/live-agent/integration/magento-plugin/frequently-asked-questions.html\' target="_blank">bit longer</a>.
39
+ </div>
40
+ <div class="nlLoader"><span>loading...</span></div>
41
+ <div class="nlDarkOrangeText" name="liveagent_wait_status" id="liveagent_wait_status">
42
+ Installing...
43
+ </div>
44
+ </div>
45
+ <form id="liveagent_wait_form" name="edit_form" action="{completeUrlAction}" method="post">
46
+ <input name="form_key" type="hidden" value="{formKey}" />
47
+ <input name="action" type="hidden" value="r"/>
48
+ </form>
49
+ <div class="nlBoxWithScreenshots"></div>
50
+ <div class="nlContactUs">
51
+ <div class="nlDarkOrangeText">
52
+ Do you need help with this plugin?
53
+ </div>
54
+ <div class="nlDarkOrangeText">
55
+ Feel free to
56
+ <a href="http://support.qualityunit.com/la/chat/index.php?bid=858b57ea" target="_blank">
57
+ contact us
58
+ </a>
59
+ </div>
60
+ </div>
61
+ <div class="clear"></div>
62
+ </div>
63
+ </form>
64
+ </div>
65
+ </div>
66
+ <div class="clear"></div>
67
+ <script type="text/javascript"><!--//--><![CDATA[//><!--
68
+ setTimeout("document.getElementById(\'liveagent_wait_status\').innerHTML = \'Initializing...\'", 10);
69
+ var timer = 3000;
70
+ var percentage = 4;
71
+ for (i=0;i<24;i++) {
72
+ setTimeout("document.getElementById(\'liveagent_wait_status\').innerHTML = \'Installing " + percentage + "% ...\'", timer);
73
+ timer+=1000;
74
+ percentage+=4;
75
+ }
76
+ setTimeout("window.location.reload()", timer);
77
+ //--><!]]></script>
78
+ ';
79
+ }
80
+ }
app/code/local/Qualityunit/Liveagent/Exception/ConnectProblem.php ADDED
@@ -0,0 +1,12 @@
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @copyright Copyright (c) 2011 Quality Unit s.r.o.
4
+ * @author Juraj Simon
5
+ * @package MgLiveAgentPlugin
6
+ * @version 1.0.0
7
+ *
8
+ * Licensed under GPL2
9
+ */
10
+
11
+ class Qualityunit_Liveagent_Exception_ConnectProblem extends Exception {}
12
+ ?>
app/code/local/Qualityunit/Liveagent/Exception/SignupFailed.php ADDED
@@ -0,0 +1,12 @@
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @copyright Copyright (c) 2011 Quality Unit s.r.o.
4
+ * @author Juraj Simon
5
+ * @package MgLiveAgentPlugin
6
+ * @version 1.0.0
7
+ *
8
+ * Licensed under GPL2
9
+ */
10
+
11
+ class Qualityunit_Liveagent_Exception_SignupFailed extends Exception {}
12
+ ?>
app/code/local/Qualityunit/Liveagent/Helper/Auth.php ADDED
@@ -0,0 +1,79 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @copyright Copyright (c) 2007 Quality Unit s.r.o.
4
+ * @author Juraj Simon
5
+ * @package WpLiveAgentPlugin
6
+ * @version 1.0.0
7
+ *
8
+ * Licensed under GPL2
9
+ */
10
+
11
+ class Qualityunit_Liveagent_Helper_Auth extends Qualityunit_Liveagent_Helper_Base {
12
+ public function ping() {
13
+ if (strpos($this->getRemoteApiUrl(), '.ladesk.com') === false) {
14
+ $this->internalPing($this->getRemoteApiUrl());
15
+ return;
16
+ }
17
+ //hack to fix dns problem - should be romved in the future myla.ladesk.com is not accessible, but www.myla.ladesk.com is - so try both
18
+ try {
19
+ $this->internalPing($this->getRemoteApiUrl());
20
+ } catch (Qualityunit_Liveagent_Exception_ConnectProblem $e) {
21
+ $url = $this->getRemoteApiUrl();
22
+ $url = preg_replace('/http:\/\//', 'http://www.', $url);
23
+ $this->internalPing($url);
24
+ }
25
+ }
26
+
27
+ private function internalPing($url) {
28
+ $request = new La_Rpc_DataRequest("Gpf_Common_ConnectionUtil", "ping");
29
+ $request->setUrl($url);
30
+ try {
31
+ $request->sendNow();
32
+ } catch (Exception $e) {
33
+ $this->_log('Unable to ping Live Agent remotelly' . $e->getMessage());
34
+ throw new Qualityunit_Liveagent_Exception_ConnectProblem();
35
+ }
36
+ $data = $request->getData();
37
+ if ($data->getParam('status') != 'OK') {
38
+ throw new Qualityunit_Liveagent_Exception_ConnectProblem();
39
+ }
40
+ }
41
+
42
+ private function InternalLoginAndGetLoginData($url) {
43
+ $request = new La_Rpc_DataRequest("Gpf_Api_AuthService", "authenticate");
44
+
45
+ $request->setField('username' ,$this->settingsModel->getOwnerEmail());
46
+ $request->setField('password' ,$this->settingsModel->getOwnerPassword());
47
+ $request->setUrl($url);
48
+ try {
49
+ $request->sendNow();
50
+ } catch (Exception $e) {
51
+ $this->_log('Unable to login.');
52
+ //$this->_log($e->getMessage());
53
+ throw new Qualityunit_Liveagent_Exception_ConnectProblem();
54
+ }
55
+ if ($request->getData()->getParam('error')!=null) {
56
+ $this->_log('Answer from server: ' . print_r($request->getResponseObject()->toObject(), true));
57
+ throw new Qualityunit_Liveagent_Exception_ConnectProblem();
58
+ }
59
+ return $request->getData();
60
+ }
61
+
62
+ /**
63
+ * @return La_Rpc_Data
64
+ */
65
+ public function LoginAndGetLoginData() {
66
+ if (strpos($this->getRemoteApiUrl(), '.ladesk.com') === false) {
67
+ return $this->InternalLoginAndGetLoginData($this->getRemoteApiUrl());
68
+ }
69
+ //hack to fix dns problem - should be romved in the future myla.ladesk.com is not accessible, but www.myla.ladesk.com is - so try both
70
+ try {
71
+ return $this->InternalLoginAndGetLoginData($this->getRemoteApiUrl());
72
+ } catch (Qualityunit_Liveagent_Exception_ConnectProblem $e) {
73
+ $url = $this->getRemoteApiUrl();
74
+ $url = preg_replace('/http:\/\//', 'http://www.', $url);
75
+ return $this->InternalLoginAndGetLoginData($url);
76
+ }
77
+ }
78
+ }
79
+ ?>
app/code/local/Qualityunit/Liveagent/Helper/Base.php ADDED
@@ -0,0 +1,52 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @copyright Copyright (c) 2007 Quality Unit s.r.o.
4
+ * @author Juraj Simon
5
+ * @package WpLiveAgentPlugin
6
+ * @version 1.0.0
7
+ *
8
+ * Licensed under GPL2
9
+ */
10
+
11
+ class Qualityunit_Liveagent_Helper_Base {
12
+ /**
13
+ * @var Qualityunit_Liveagent_Helper_Settings
14
+ */
15
+ protected $settingsModel;
16
+ const ACCOUNT_STATUS_NOTSET = 'N';
17
+ const ACCOUNT_STATUS_SET = 'S';
18
+ const ACCOUNT_STATUS_WAIT = 'W';
19
+
20
+ public function _log($message) {
21
+ Mage::log($message, Zend_Log::DEBUG);
22
+ }
23
+
24
+ public function __construct() {
25
+ $this->settingsModel = new Qualityunit_Liveagent_Helper_Settings();
26
+ }
27
+
28
+ public function showAdminError($error) {
29
+ Mage::log($message, Zend_Log::ERR);
30
+ }
31
+
32
+ public function showConnectionError() {
33
+ $this->showAdminError(Mage::helper('liveagent')->__('Unable to connect to Live Agent. please check your connection settings'));
34
+ }
35
+
36
+ public function getRemoteTrackJsUrl() {
37
+ return $this->settingsModel->getOption(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME) . '/scripts/trackjs.php';
38
+ }
39
+
40
+ public function getRemotePixUrl() {
41
+ return $this->settingsModel->getOption(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME) . '/scripts/pix.gif';
42
+ }
43
+
44
+ public function getRemoteApiUrl() {
45
+ return $this->settingsModel->getOption(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME) . '/api/index.php';
46
+ }
47
+
48
+ protected function isEmpty($var) {
49
+ return $var=== null || $var=='';
50
+ }
51
+ }
52
+ ?>
app/code/local/Qualityunit/Liveagent/Helper/Buttons.php ADDED
@@ -0,0 +1,143 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @copyright Copyright (c) 2007 Quality Unit s.r.o.
4
+ * @author Juraj Simon
5
+ * @package WpLiveAgentPlugin
6
+ * @version 1.0.0
7
+ *
8
+ * Licensed under GPL2
9
+ */
10
+
11
+ class Qualityunit_Liveagent_Helper_Buttons extends Qualityunit_Liveagent_Helper_Base {
12
+
13
+ const CLASS_NAME = 'La_Button_ButtonTable';
14
+
15
+ public function refresh() {
16
+ $rows = $this->getData();
17
+ if ($rows->getSize() == 0) {
18
+ Mage::log('No buttons recieved from LA', Zend_Log::INFO);
19
+ return;
20
+ }
21
+ Mage::log($rows->getSize() . ' buttons recieved from LA', Zend_Log::INFO);
22
+ Mage::log(print_r($rows, true), Zend_Log::INFO);
23
+ $this->settingsModel->updateLastButtonsReloadTime();
24
+ //clear existing buttons
25
+ Mage::log('Clearing cached buttons...', Zend_Log::INFO);
26
+ $enabledButtons = array();
27
+ foreach (Mage::getModel('liveagent/buttons')->getCollection() as $row) {
28
+ Mage::log('Clearing button ' . $row->getButtonid() , Zend_Log::INFO);
29
+ if ($row->getOnlinestatus() == 'Y') {
30
+ $enabledButtons[] = $row->getButtonid();
31
+ }
32
+ $row->delete();
33
+ }
34
+ Mage::log('Clear complete', Zend_Log::INFO);
35
+ foreach ($rows as $row) {
36
+ Mage::log('Processing button ' . $row->get('id'), Zend_Log::INFO);
37
+ $button = Mage::getModel('liveagent/buttons')->load($row->get('id'), 'buttonid');
38
+ $button->setButtonid($row->get('id'));
39
+ $button->setUserid($row->get('userid'));
40
+ $button->setName($row->get('name'));
41
+ $button->setContenttype($row->get('contenttype'));
42
+ $button->setOnlinecode($row->get('onlinecode'));
43
+ $button->setOfflinecode($row->get('offlinecode'));
44
+ $button->setImpressions($row->get('impressions'));
45
+ $button->setChats($row->get('chats'));
46
+ $button->setData1($row->get('data1'));
47
+ $button->setData2($row->get('data2'));
48
+ $button->setData3($row->get('data3'));
49
+ $button->setData4($row->get('data4'));
50
+ $button->setData5($row->get('data5'));
51
+ if (in_array($row->get('id'), $enabledButtons)) {
52
+ $button->setOnlinestatus('Y');
53
+ } else {
54
+ $button->setOnlinestatus('N');
55
+ }
56
+ $button->save();
57
+ }
58
+ }
59
+
60
+ public function enableDefaultButtonIfNoImpressions() {
61
+ //b2222222 only for compatibility reasons
62
+ $button1 = Mage::getModel('liveagent/buttons')->load('b2222222', 'buttonid');
63
+ if ($button1->getButtonid() != '' && $button1->getImpressions() > 0) {
64
+ $this->enableDefaultButton();
65
+ }
66
+ $button2 = Mage::getModel('liveagent/buttons')->load('button1', 'buttonid');
67
+ if($button2->getButtonid() != '' && $button2->getImpressions() > 0) {
68
+ $this->enableDefaultButton('button1');
69
+ }
70
+ }
71
+
72
+ public function enableDefaultButton($buttonId = 'b2222222') {
73
+ //b2222222 only for compatibility reasons
74
+ $button = Mage::getModel('liveagent/buttons')->load($buttonId, 'buttonid');
75
+ if ($button->getButtonid() != '') {
76
+ $button->setOnlinestatus('Y');
77
+ $button->save();
78
+ }
79
+ }
80
+
81
+ public function getIntegrationCode($buttonId) {
82
+ return '<img src="'.$this->getRemotePixUrl().'" onLoad="LiveAgentTracker.createButton(\''.$buttonId.'\', this);"/>' . "\n";
83
+ }
84
+
85
+ public function setOption($name, $value) {
86
+ $setting = Mage::getModel('liveagent/settings')->load($name, 'name');
87
+ $setting->setValue($value);
88
+ $setting->setName($name);
89
+ $setting->save();
90
+ }
91
+
92
+ public function getOption($name) {
93
+ $setting = Mage::getModel('liveagent/settings')->load($name, 'name');
94
+ return $setting->getData('value');
95
+ }
96
+
97
+ public function getAccountStatus() {
98
+ return $this->getOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_STATUS);
99
+ }
100
+
101
+ public function connectionSettingsAreEmpty() {
102
+ return $this->getOption(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME) == '' && $this->getOption(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME) == '';
103
+ }
104
+
105
+ private function internalGetData($url) {
106
+ $request = new La_Rpc_Request(self::CLASS_NAME, 'getRows');
107
+ try {
108
+ $request->setUrl($url);
109
+ } catch (Qualityunit_Liveagent_Exception_ConnectProblem $e) {
110
+ $this->_log('Unable to connect and get session id');
111
+ return new La_Data_RecordSet();
112
+ }
113
+
114
+ try {
115
+ $request->sendNow();
116
+ } catch (Exception $e) {
117
+ $this->_log(sprintf('Unable to get data for %s', self::CLASS_NAME));
118
+ $this->_log($e->getMessage());
119
+ return new La_Data_RecordSet();
120
+ }
121
+ $grid = new La_Data_Grid();
122
+ $grid->loadFromObject($request->getStdResponse());
123
+ return $grid->getRecordset();
124
+ }
125
+
126
+ public function getData() {
127
+ $url = $this->getRemoteApiUrl() . '?S=' . $this->settingsModel->getOwnerSessionId();
128
+ if (strpos($this->getRemoteApiUrl(), '.ladesk.com') === false) {
129
+ return $this->internalGetData($url);
130
+ }
131
+ $secondUrl = preg_replace('/http:\/\//', 'http://www.', $url);
132
+ try {
133
+ $rows = $this->internalGetData($url);
134
+ } catch (Exception $e) {
135
+ return $this->internalGetData($secondUrl);
136
+ }
137
+ if ($rows->getSize() == 0) {
138
+ return $this->internalGetData($secondUrl);
139
+ }
140
+ return $rows;
141
+ }
142
+ }
143
+ ?>
app/code/local/Qualityunit/Liveagent/Helper/Data.php ADDED
@@ -0,0 +1,6 @@
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Helper_Data extends Mage_Core_Helper_Abstract
4
+ {
5
+
6
+ }
app/code/local/Qualityunit/Liveagent/Helper/PhpApi.php ADDED
@@ -0,0 +1,4310 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @copyright Copyright (c) 2010-2011 Quality Unit s.r.o.
4
+ * @author Quality Unit
5
+ * @package PhpApi
6
+ *
7
+ * Licensed under the Quality Unit, s.r.o. Dual License Agreement,
8
+ * Version 1.0 (the "License"); you may not use this file except in compliance
9
+ * with the License. You may obtain a copy of the License at
10
+ * http://www.qualityunit.com/licenses/gpf
11
+ * Generated on: 2011-06-06 04:17:42
12
+ * Framework version: 1.5.22
13
+ *
14
+ */
15
+
16
+ @ini_set('session.gc_maxlifetime', 28800);
17
+ @ini_set('session.cookie_path', '/');
18
+ @ini_set('session.use_cookies', true);
19
+ @ini_set('magic_quotes_runtime', false);
20
+ @ini_set('session.use_trans_sid', false);
21
+ @ini_set('zend.ze1_compatibility_mode', false);
22
+
23
+ if (!class_exists('La_Lang', false)) {
24
+ class La_Lang {
25
+ public static function _replaceArgs($message, $args = null) {
26
+ if (!is_array($args)) {
27
+ $args = func_get_args();
28
+ }
29
+ if(count($args) > 1 && substr_count($message, '%s') < count($args)) {
30
+ array_shift($args);
31
+ return vsprintf($message, $args);
32
+ }
33
+ return $message;
34
+ }
35
+
36
+ public static function _($message, $args = null, $langCode = '') {
37
+ if (!is_array($args)) {
38
+ $args = func_get_args();
39
+ }
40
+ return self::_replaceArgs($message, $args);
41
+ }
42
+
43
+ public static function _sys($message, $args = null) {
44
+ if (!is_array($args)) {
45
+ $args = func_get_args();
46
+ }
47
+ return self::_replaceArgs($message, $args);
48
+ }
49
+
50
+ public static function _runtime($message) {
51
+ return $message;
52
+ }
53
+
54
+ public static function _localizeRuntime($message, $langCode = '') {
55
+ preg_match_all('/##(.+?)##/ms', $message, $attributes, PREG_OFFSET_CAPTURE);
56
+ foreach ($attributes[1] as $index => $attribute) {
57
+ $message = str_replace($attributes[0][$index][0], self::_($attribute[0], null, $langCode), $message);
58
+ }
59
+ return $message;
60
+
61
+ }
62
+ }
63
+ }
64
+
65
+ if (!class_exists('La_Object', false)) {
66
+ class La_Object {
67
+ /**
68
+ * Translate input message into selected language.
69
+ * If translation will not be found, return same message.
70
+ *
71
+ * @param string $message
72
+ * @return string
73
+ */
74
+ public function _($message) {
75
+ $args = func_get_args();
76
+ return La_Lang::_($message, $args);
77
+ }
78
+
79
+ /**
80
+ * Translates text enclosed in ##any text##
81
+ * This function is not parsed by language parser, because as input should be used e.g. texts loaded from database
82
+ *
83
+ * @param string $message String to translate
84
+ * @return string Translated text
85
+ */
86
+ public function _localize($message) {
87
+ return La_Lang::_localizeRuntime($message);
88
+ }
89
+
90
+ /**
91
+ * Translate input message into default language defined in language settings for account.
92
+ * This function should be used in case message should be translated to default language (e.g. log messages written to event log)
93
+ *
94
+ * @param string $message
95
+ * @return string
96
+ */
97
+ public function _sys($message) {
98
+ $args = func_get_args();
99
+ return La_Lang::_sys($message, $args);
100
+ }
101
+ }
102
+
103
+ } //end La_Object
104
+
105
+ if (!interface_exists('La_Controller', false)) {
106
+ interface La_Controller {
107
+ /**
108
+ * @throws La_Controller_Exception_UnsupportedRequest
109
+ */
110
+ public function execute();
111
+ }
112
+
113
+ } //end La_Controller
114
+
115
+ if (!interface_exists('La_Rpc_Serializable', false)) {
116
+ interface La_Rpc_Serializable {
117
+
118
+ public function toObject();
119
+
120
+ public function toText();
121
+ }
122
+
123
+ } //end La_Rpc_Serializable
124
+
125
+ if (!interface_exists('La_Rpc_DataEncoder', false)) {
126
+ interface La_Rpc_DataEncoder {
127
+ function encodeResponse(La_Rpc_Serializable $response);
128
+ }
129
+
130
+
131
+
132
+ } //end La_Rpc_DataEncoder
133
+
134
+ if (!interface_exists('La_Rpc_DataDecoder', false)) {
135
+ interface La_Rpc_DataDecoder {
136
+ /**
137
+ * @param string $str
138
+ * @return StdClass
139
+ */
140
+ function decode($str);
141
+ }
142
+
143
+
144
+
145
+ } //end La_Rpc_DataDecoder
146
+
147
+ if (!class_exists('La_Rpc_Array', false)) {
148
+ class La_Rpc_Array extends La_Object implements La_Rpc_Serializable, IteratorAggregate {
149
+
150
+ private $array;
151
+
152
+ function __construct(array $array = null){
153
+ if($array === null){
154
+ $this->array = array();
155
+ }else{
156
+ $this->array = $array;
157
+ }
158
+ }
159
+
160
+ public function add($response) {
161
+ if(is_scalar($response) || $response instanceof La_Rpc_Serializable) {
162
+ $this->array[] = $response;
163
+ return;
164
+ }
165
+ throw new La_Exception("Value of type " . gettype($response) . " is not scalar or La_Rpc_Serializable");
166
+ }
167
+
168
+ public function toObject() {
169
+ $array = array();
170
+ foreach ($this->array as $response) {
171
+ if($response instanceof La_Rpc_Serializable) {
172
+ $array[] = $response->toObject();
173
+ } else {
174
+ $array[] = $response;
175
+ }
176
+ }
177
+ return $array;
178
+ }
179
+
180
+ public function toText() {
181
+ return var_dump($this->array);
182
+ }
183
+
184
+ public function getCount() {
185
+ return count($this->array);
186
+ }
187
+
188
+ public function get($index) {
189
+ return $this->array[$index];
190
+ }
191
+
192
+ /**
193
+ *
194
+ * @return ArrayIterator
195
+ */
196
+ public function getIterator() {
197
+ return new ArrayIterator($this->array);
198
+ }
199
+ }
200
+
201
+ } //end La_Rpc_Array
202
+
203
+ if (!class_exists('La_Rpc_Server', false)) {
204
+ class La_Rpc_Server extends La_Object implements La_Controller {
205
+ const REQUESTS = 'R';
206
+ const RUN_METHOD = 'run';
207
+ const FORM_REQUEST = 'FormRequest';
208
+ const FORM_RESPONSE = 'FormResponse';
209
+ const BODY_DATA_NAME = 'D';
210
+
211
+
212
+ const HANDLER_FORM = 'Y';
213
+ const HANDLER_JASON = 'N';
214
+ const HANDLER_WINDOW_NAME = 'W';
215
+
216
+ /**
217
+ * @var La_Rpc_DataEncoder
218
+ */
219
+ private $dataEncoder;
220
+ /**
221
+ * @var La_Rpc_DataDecoder
222
+ */
223
+ private $dataDecoder;
224
+
225
+ public function __construct() {
226
+ }
227
+
228
+ private function initDatabaseLogger() {
229
+ $logger = La_Log_Logger::getInstance();
230
+
231
+ if(!$logger->checkLoggerTypeExists(La_Log_LoggerDatabase::TYPE)) {
232
+ $logger->setGroup(La_Common_String::generateId(10));
233
+ $logLevel = La_Settings::get(La_Settings_Gpf::LOG_LEVEL_SETTING_NAME);
234
+ $logger->add(La_Log_LoggerDatabase::TYPE, $logLevel);
235
+ }
236
+ }
237
+
238
+ /**
239
+ * Return response to standard output
240
+ */
241
+ public function execute($request = '') {
242
+ $response = $this->encodeResponse($this->executeRequest($request));
243
+ La_Http::output($response);
244
+ }
245
+
246
+ /**
247
+ * @return La_Rpc_Serializable
248
+ */
249
+ public function executeRequest($request = '') {
250
+ try {
251
+ if(isset($_REQUEST[self::BODY_DATA_NAME])) {
252
+ $request = $this->parseRequestDataFromPost($_REQUEST[self::BODY_DATA_NAME]);
253
+ }
254
+ if($this->isStandardRequestUsed($_REQUEST)) {
255
+ $request = $this->setStandardRequest();
256
+ }
257
+
258
+ $this->setDecoder($request);
259
+ $params = new La_Rpc_Params($this->decodeRequest($request));
260
+ if ($params->getClass() == '' || $params->getMethod() == '') {
261
+ throw new La_Controller_Exception_UnsupportedRequest();
262
+ }
263
+ $this->setEncoder($params);
264
+ $response = $this->executeRequestParams($params);
265
+ } catch (La_Controller_Exception_UnsupportedRequest $e) {
266
+ throw $e;
267
+ } catch (Exception $e) {
268
+ return new La_Rpc_ExceptionResponse($e);
269
+ }
270
+ return $response;
271
+ }
272
+
273
+ private function parseRequestDataFromPost($data) {
274
+ if(get_magic_quotes_gpc()) {
275
+ return stripslashes($data);
276
+ }
277
+ return $data;
278
+ }
279
+
280
+ /**
281
+ *
282
+ * @param unknown_type $requestObj
283
+ * @return La_Rpc_Serializable
284
+ */
285
+ private function executeRequestParams(La_Rpc_Params $params) {
286
+ if (La_Application::getInstance()->isInMaintenanceMode()
287
+ && !La_Paths::getInstance()->isInstallModeActive()) {
288
+ return new La_Rpc_MaintenenceModeResponse();
289
+ }
290
+ try {
291
+ La_Db_LoginHistory::logRequest();
292
+ return $this->callServiceMethod($params);
293
+ } catch (La_Session_Exception_SessionExpired $e) {
294
+ return new La_Rpc_SessionExpiredResponse($e);
295
+ } catch (Exception $e) {
296
+ return new La_Rpc_ExceptionResponse($e);
297
+ }
298
+ }
299
+
300
+ /**
301
+ * @throws La_Session_Exception_SessionExpired
302
+ */
303
+ protected function callServiceMethod(La_Rpc_Params $params) {
304
+ $method = new La_Rpc_ServiceMethod($params);
305
+ return $method->invoke($params);
306
+ }
307
+
308
+ /**
309
+ * Compute correct handler type for server response
310
+ *
311
+ * @param array $requestData
312
+ * @param string $type
313
+ * @return string
314
+ */
315
+ private function getEncoderHandlerType($requestData) {
316
+ if ($this->isFormHandler($requestData, self::FORM_RESPONSE, self::HANDLER_FORM)) {
317
+ return self::HANDLER_FORM;
318
+ }
319
+ if ($this->isFormHandler($requestData, self::FORM_RESPONSE, self::HANDLER_WINDOW_NAME)) {
320
+ return self::HANDLER_WINDOW_NAME;
321
+ }
322
+ return self::HANDLER_JASON;
323
+ }
324
+
325
+
326
+ private function isFormHandler($requestData, $type, $handler) {
327
+ return (isset($_REQUEST[$type]) && $_REQUEST[$type] == $handler) ||
328
+ (isset($requestData) && isset($requestData[$type]) && $requestData[$type] == $handler);
329
+ }
330
+
331
+ private function decodeRequest($requestData) {
332
+ return $this->dataDecoder->decode($requestData);
333
+ }
334
+
335
+ private function isStandardRequestUsed($requestArray) {
336
+ return is_array($requestArray) && array_key_exists(La_Rpc_Params::CLASS_NAME, $requestArray);
337
+ }
338
+
339
+ private function setStandardRequest() {
340
+ return array_merge($_POST, $_GET);
341
+ }
342
+
343
+ private function isFormRequest($request) {
344
+ return $this->isFormHandler($request, self::FORM_REQUEST, self::HANDLER_FORM);
345
+ }
346
+
347
+ private function encodeResponse(La_Rpc_Serializable $response) {
348
+ return $this->dataEncoder->encodeResponse($response);
349
+ }
350
+
351
+
352
+ private function setDecoder($request) {
353
+ if ($this->isFormRequest($request)) {
354
+ $this->dataDecoder = new La_Rpc_FormHandler();
355
+ } else {
356
+ $this->dataDecoder = new La_Rpc_Json();
357
+ }
358
+ }
359
+
360
+ private function setEncoder(La_Rpc_Params $params) {
361
+ switch ($params->get(self::FORM_RESPONSE)) {
362
+ case self::HANDLER_FORM:
363
+ $this->dataEncoder = new La_Rpc_FormHandler();
364
+ break;
365
+ case self::HANDLER_WINDOW_NAME:
366
+ $this->dataEncoder = new La_Rpc_WindowNameHandler();
367
+ break;
368
+ default:
369
+ $this->dataEncoder = new La_Rpc_Json();
370
+ break;
371
+ }
372
+ }
373
+
374
+ /**
375
+ * Executes multi request
376
+ *
377
+ * @service
378
+ * @anonym
379
+ * @return La_Rpc_Serializable
380
+ */
381
+ public function run(La_Rpc_Params $params) {
382
+ $requestArray = $params->get(self::REQUESTS);
383
+
384
+ $response = new La_Rpc_Array();
385
+ foreach ($requestArray as $request) {
386
+ $response->add($this->executeRequestParams(new La_Rpc_Params($request)));
387
+ }
388
+ return $response;
389
+ }
390
+
391
+ /**
392
+ * Set time offset between client and server and store it to session
393
+ * Offset is computed as client time - server time
394
+ *
395
+ * @anonym
396
+ * @service
397
+ * @param La_Rpc_Params $params
398
+ * @return La_Rpc_Action
399
+ */
400
+ public function syncTime(La_Rpc_Params $params) {
401
+ $action = new La_Rpc_Action($params);
402
+ La_Module::getProperties()->setTimeOffset($action->getParam('offset')/1000);
403
+ $action->addOk();
404
+ return $action;
405
+ }
406
+ }
407
+
408
+ } //end La_Rpc_Server
409
+
410
+ if (!class_exists('La_Rpc_MultiRequest', false)) {
411
+ class La_Rpc_MultiRequest extends La_Object {
412
+ private $url = '';
413
+ /**
414
+ *
415
+ * @var La_Rpc_Array
416
+ */
417
+ private $requests;
418
+ /**
419
+ * @var La_Rpc_Json
420
+ */
421
+ private $json;
422
+ protected $serverClassName = 'Gpf_Rpc_Server';
423
+
424
+ private $sessionId = null;
425
+
426
+ private $debugRequests = false;
427
+
428
+ /**
429
+ * @var La_Rpc_MultiRequest
430
+ */
431
+ private static $instance;
432
+
433
+ public function __construct() {
434
+ $this->json = new La_Rpc_Json();
435
+ $this->requests = new La_Rpc_Array();
436
+ }
437
+
438
+ /**
439
+ * @return La_Rpc_MultiRequest
440
+ */
441
+ public static function getInstance() {
442
+ if(self::$instance === null) {
443
+ self::$instance = new La_Rpc_MultiRequest();
444
+ }
445
+ return self::$instance;
446
+ }
447
+
448
+ public static function setInstance(La_Rpc_MultiRequest $instance) {
449
+ self::$instance = $instance;
450
+ }
451
+
452
+ public function add(La_Rpc_Request $request) {
453
+ $this->requests->add($request);
454
+ }
455
+
456
+ protected function sendRequest($requestBody) {
457
+ $request = new La_Net_Http_Request();
458
+
459
+ $request->setMethod('POST');
460
+ $request->setBody(La_Rpc_Server::BODY_DATA_NAME . '=' . urlencode($requestBody));
461
+ $request->setUrl($this->url);
462
+
463
+ $client = new La_Net_Http_Client();
464
+ $response = $client->execute($request);
465
+ return $response->getBody();
466
+ }
467
+
468
+ public function setSessionId($sessionId) {
469
+ $this->sessionId = $sessionId;
470
+ }
471
+
472
+ public function setDebugRequests($debug) {
473
+ $this->debugRequests = $debug;
474
+ }
475
+
476
+ public function send() {
477
+ $request = new La_Rpc_Request($this->serverClassName, La_Rpc_Server::RUN_METHOD);
478
+ $request->addParam(La_Rpc_Server::REQUESTS, $this->requests);
479
+ if($this->sessionId != null) {
480
+ $request->addParam("S", $this->sessionId);
481
+ }
482
+ $requestBody = $this->json->encodeResponse($request);
483
+ $responseText = $this->sendRequest($requestBody);
484
+ $responseArray = $this->json->decode($responseText);
485
+ if (!is_array($responseArray)) {
486
+ throw new La_Exception("Response decoding failed: not array. Received text: $responseText");
487
+ }
488
+
489
+ if (count($responseArray) != $this->requests->getCount()) {
490
+ throw new La_Exception("Response decoding failed: Number of responses is not same as number of requests");
491
+ }
492
+
493
+ $exception = false;
494
+ foreach ($responseArray as $index => $response) {
495
+ if (is_object($response) && isset($response->e)) {
496
+ $exception = true;
497
+ $this->requests->get($index)->setResponseError($response->e);
498
+ } else {
499
+ $this->requests->get($index)->setResponse($response);
500
+ }
501
+ }
502
+ if($exception) {
503
+ $messages = '';
504
+ foreach ($this->requests as $request) {
505
+ $messages .= $request->getResponseError() . "|";
506
+ }
507
+ }
508
+ $this->requests = new La_Rpc_Array();
509
+ if($exception) {
510
+ throw new La_Rpc_ExecutionException($messages);
511
+ }
512
+ }
513
+
514
+ public function setUrl($url) {
515
+ $this->url = $url;
516
+ }
517
+
518
+ public function getUrl() {
519
+ return $this->url;
520
+ }
521
+
522
+ private function getCookies() {
523
+ $cookiesString = '';
524
+ foreach ($_COOKIE as $name => $value) {
525
+ $cookiesString .= "$name=$value;";
526
+ }
527
+ return $cookiesString;
528
+ }
529
+ }
530
+
531
+
532
+ } //end La_Rpc_MultiRequest
533
+
534
+ if (!class_exists('La_Rpc_Params', false)) {
535
+ class La_Rpc_Params extends La_Object implements La_Rpc_Serializable {
536
+ private $params;
537
+ const CLASS_NAME = 'C';
538
+ const METHOD_NAME = 'M';
539
+ const SESSION_ID = 'S';
540
+ const ACCOUNT_ID = 'aid';
541
+
542
+ function __construct($params = null) {
543
+ if($params === null) {
544
+ $this->params = new stdClass();
545
+ return;
546
+ }
547
+ $this->params = $params;
548
+ }
549
+
550
+ public static function createGetRequest($className, $methodName = 'execute', $formRequest = false, $formResponse = false) {
551
+ $requestData = array();
552
+ $requestData[self::CLASS_NAME] = $className;
553
+ $requestData[self::METHOD_NAME] = $methodName;
554
+ $requestData[La_Rpc_Server::FORM_REQUEST] = $formRequest ? Gpf::YES : '';
555
+ $requestData[La_Rpc_Server::FORM_RESPONSE] = $formResponse ? Gpf::YES : '';
556
+ return $requestData;
557
+ }
558
+
559
+ /**
560
+ *
561
+ * @param unknown_type $className
562
+ * @param unknown_type $methodName
563
+ * @param unknown_type $formRequest
564
+ * @param unknown_type $formResponse
565
+ * @return La_Rpc_Params
566
+ */
567
+ public static function create($className, $methodName = 'execute', $formRequest = false, $formResponse = false) {
568
+ $params = new La_Rpc_Params();
569
+ $obj = new stdClass();
570
+ foreach (self::createGetRequest($className, $methodName, $formRequest, $formResponse) as $name => $value) {
571
+ $params->add($name,$value);
572
+ }
573
+ return $params;
574
+ }
575
+
576
+ public function setArrayParams(array $params) {
577
+ foreach ($params as $name => $value) {
578
+ $this->add($name, $value);
579
+ }
580
+ }
581
+
582
+ public function exists($name) {
583
+ if(!is_object($this->params) || !array_key_exists($name, $this->params)) {
584
+ return false;
585
+ }
586
+ return true;
587
+ }
588
+
589
+ /**
590
+ *
591
+ * @param unknown_type $name
592
+ * @return mixed Return null if $name does not exist.
593
+ */
594
+ public function get($name) {
595
+ if(!$this->exists($name)) {
596
+ return null;
597
+ }
598
+ return $this->params->{$name};
599
+ }
600
+
601
+ public function set($name, $value) {
602
+ if(!$this->exists($name)) {
603
+ return;
604
+ }
605
+ $this->params->{$name} = $value;
606
+ }
607
+
608
+ public function add($name, $value) {
609
+ $this->params->{$name} = $value;
610
+ }
611
+
612
+ public function getClass() {
613
+ return $this->get(self::CLASS_NAME);
614
+ }
615
+
616
+ public function getMethod() {
617
+ return $this->get(self::METHOD_NAME);
618
+ }
619
+
620
+ public function getSessionId() {
621
+ return $this->get(self::SESSION_ID);
622
+ }
623
+
624
+ public function clearSessionId() {
625
+ $this->set(self::SESSION_ID, null);
626
+ }
627
+
628
+ public function getAccountId() {
629
+ return $this->get(self::ACCOUNT_ID);
630
+ }
631
+
632
+ public function toObject() {
633
+ return $this->params;
634
+ }
635
+
636
+ public function toText() {
637
+ throw new La_Exception("Unimplemented");
638
+ }
639
+ }
640
+
641
+
642
+ } //end La_Rpc_Params
643
+
644
+ if (!class_exists('La_Exception', false)) {
645
+ class La_Exception extends Exception {
646
+
647
+ private $id;
648
+
649
+ public function __construct($message = '',$code = null) {
650
+ $trace = '';
651
+ foreach (debug_backtrace(false) as $i => $traceStep) {
652
+ $trace .= sprintf("#%s - %s::%s() at line %s<br>\n", $i, @$traceStep['class'], @$traceStep['function'], @$traceStep['line']);
653
+ }
654
+ $message .= "<br>\nTRACE:<br>\n" . $trace;
655
+ parent::__construct($message, $code);
656
+ }
657
+
658
+ protected function logException() {
659
+ La_Log::error($this->getMessage());
660
+ }
661
+
662
+ public function setId($id) {
663
+ $this->id = $id;
664
+ }
665
+
666
+ public function getId() {
667
+ return $this->id;
668
+ }
669
+
670
+ }
671
+
672
+ } //end La_Exception
673
+
674
+ if (!class_exists('La_Data_RecordSetNoRowException', false)) {
675
+ class La_Data_RecordSetNoRowException extends La_Exception {
676
+ public function __construct($keyValue) {
677
+ parent::__construct("'Row $keyValue does not exist");
678
+ }
679
+
680
+ protected function logException() {
681
+ }
682
+ }
683
+
684
+ } //end La_Data_RecordSetNoRowException
685
+
686
+ if (!class_exists('La_Rpc_ExecutionException', false)) {
687
+ class La_Rpc_ExecutionException extends La_Exception {
688
+
689
+ function __construct($message) {
690
+ parent::__construct('RPC Execution exception: ' . $message);
691
+ }
692
+ }
693
+
694
+ } //end La_Rpc_ExecutionException
695
+
696
+ if (!class_exists('La_Rpc_Object', false)) {
697
+ class La_Rpc_Object extends La_Object implements La_Rpc_Serializable {
698
+
699
+ private $object;
700
+
701
+ public function __construct($object = null) {
702
+ $this->object = $object;
703
+ }
704
+
705
+ public function toObject() {
706
+ if ($this->object != null) {
707
+ return $this->object;
708
+ }
709
+ return $this;
710
+ }
711
+
712
+ public function toText() {
713
+ return var_dump($this);
714
+ }
715
+ }
716
+
717
+
718
+ } //end La_Rpc_Object
719
+
720
+ if (!class_exists('La_Rpc_Request', false)) {
721
+ class La_Rpc_Request extends La_Object implements La_Rpc_Serializable {
722
+ protected $className;
723
+ protected $methodName;
724
+ private $responseError;
725
+ protected $response;
726
+ protected $apiSessionObject = null;
727
+
728
+ /**
729
+ * @var La_Rpc_MultiRequest
730
+ */
731
+ private $multiRequest;
732
+
733
+ /**
734
+ * @var La_Rpc_Params
735
+ */
736
+ protected $params;
737
+ private $accountId = null;
738
+
739
+ public function __construct($className, $methodName, La_Api_Session $apiSessionObject = null) {
740
+ $this->className = $className;
741
+ $this->methodName = $methodName;
742
+ $this->params = new La_Rpc_Params();
743
+ $this->setRequiredParams($this->className, $this->methodName);
744
+ if($apiSessionObject != null) {
745
+ $this->apiSessionObject = $apiSessionObject;
746
+ }
747
+ }
748
+
749
+ public function setAccountId($accountId) {
750
+ $this->accountId = $accountId;
751
+ }
752
+
753
+ public function addParam($name, $value) {
754
+ if(is_scalar($value) || is_null($value)) {
755
+ $this->params->add($name, $value);
756
+ return;
757
+ }
758
+ if($value instanceof La_Rpc_Serializable) {
759
+ $this->params->add($name, $value->toObject());
760
+ return;
761
+ }
762
+ throw new La_Exception("Cannot add request param: Value ($name=$value) is not scalar or La_Rpc_Serializable");
763
+ }
764
+
765
+ /**
766
+ *
767
+ * @return La_Rpc_MultiRequest
768
+ */
769
+ private function getMultiRequest() {
770
+ if($this->multiRequest === null) {
771
+ return La_Rpc_MultiRequest::getInstance();
772
+ }
773
+ return $this->multiRequest;
774
+ }
775
+
776
+ public function setUrl($url) {
777
+ $this->multiRequest = new La_Rpc_MultiRequest();
778
+ $this->multiRequest->setUrl($url);
779
+ }
780
+
781
+ public function send() {
782
+ if($this->apiSessionObject != null) {
783
+ $this->multiRequest = new La_Rpc_MultiRequest();
784
+ $this->multiRequest->setUrl($this->apiSessionObject->getUrl());
785
+ $this->multiRequest->setSessionId($this->apiSessionObject->getSessionId());
786
+ $this->multiRequest->setDebugRequests($this->apiSessionObject->getDebug());
787
+ }
788
+
789
+ $multiRequest = $this->getMultiRequest();
790
+ $multiRequest->add($this);
791
+ }
792
+
793
+ public function sendNow() {
794
+ $this->send();
795
+ $this->getMultiRequest()->send();
796
+ }
797
+
798
+ public function setResponseError($message) {
799
+ $this->responseError = $message;
800
+ }
801
+
802
+ public function getResponseError() {
803
+ return $this->responseError;
804
+ }
805
+
806
+ public function setResponse($response) {
807
+ $this->response = $response;
808
+ }
809
+
810
+ public function toObject() {
811
+ return $this->params->toObject();
812
+ }
813
+
814
+ public function toText() {
815
+ throw new La_Exception("Unimplemented");
816
+ }
817
+
818
+ /**
819
+ *
820
+ * @return stdClass
821
+ */
822
+ final public function getStdResponse() {
823
+ if(isset($this->responseError)) {
824
+ throw new La_Rpc_ExecutionException($this->responseError);
825
+ }
826
+ if($this->response === null) {
827
+ throw new La_Exception("Request not executed yet.");
828
+ }
829
+ return $this->response;
830
+ }
831
+
832
+ final public function getResponseObject() {
833
+ return new La_Rpc_Object($this->getStdResponse());
834
+ }
835
+
836
+ private function setRequiredParams($className, $methodName) {
837
+ $this->addParam(La_Rpc_Params::CLASS_NAME, $className);
838
+ $this->addParam(La_Rpc_Params::METHOD_NAME, $methodName);
839
+ }
840
+
841
+ /**
842
+ * @param La_Rpc_Params $params
843
+ */
844
+ public function setParams(La_Rpc_Params $params) {
845
+ $originalParams = $this->params;
846
+ $this->params = $params;
847
+ $this->setRequiredParams($originalParams->getClass(), $originalParams->getMethod());
848
+ }
849
+ }
850
+
851
+
852
+ } //end La_Rpc_Request
853
+
854
+ if (!interface_exists('La_HttpResponse', false)) {
855
+ interface La_HttpResponse {
856
+ public function setCookieValue($name, $value = null, $expire = null, $path = null, $domain = null, $secure = null, $httpOnly = null);
857
+
858
+ public function setHeaderValue($name, $value, $replace = true, $httpResponseCode = null);
859
+
860
+ public function outputText($text);
861
+ }
862
+
863
+ } //end La_HttpResponse
864
+
865
+ if (!class_exists('La_Http', false)) {
866
+ class La_Http extends La_Object implements La_HttpResponse {
867
+ /**
868
+ *
869
+ * @var La_HttpResponse
870
+ */
871
+ private static $instance = null;
872
+
873
+ /**
874
+ * @return La_Http
875
+ */
876
+ private static function getInstance() {
877
+ if(self::$instance === null) {
878
+ self::$instance = new La_Http();
879
+ }
880
+ return self::$instance;
881
+ }
882
+
883
+ public static function setInstance(La_HttpResponse $instance) {
884
+ self::$instance = $instance;
885
+ }
886
+
887
+ public static function setCookie($name, $value = null, $expire = null, $path = null, $domain = null, $secure = null, $httpOnly = null) {
888
+ self::getInstance()->setCookieValue($name, $value, $expire, $path, $domain, $secure, $httpOnly);
889
+ }
890
+
891
+ public static function setHeader($name, $value, $httpResponseCode = null) {
892
+ self::getInstance()->setHeaderValue($name, $value, true, $httpResponseCode);
893
+ }
894
+
895
+ public static function output($text) {
896
+ self::getInstance()->outputText($text);
897
+ }
898
+
899
+ public function outputText($text) {
900
+ echo $text;
901
+ }
902
+
903
+ public function setHeaderValue($name, $value, $replace = true, $httpResponseCode = null) {
904
+ $fileName = '';
905
+ $line = '';
906
+ if(headers_sent($fileName, $line)) {
907
+ throw new La_Exception("Headers already sent in $fileName line $line while setting header $name: $value");
908
+ }
909
+ header($name . ': ' . $value, $replace, $httpResponseCode);
910
+ }
911
+
912
+ public function setCookieValue($name, $value = null, $expire = null, $path = null, $domain = null, $secure = null, $httpOnly = null) {
913
+ setcookie($name, $value, $expire, $path, $domain, $secure, $httpOnly);
914
+ }
915
+
916
+ public static function getCookie($name) {
917
+ if (!array_key_exists($name, $_COOKIE)) {
918
+ return null;
919
+ }
920
+ return $_COOKIE[$name];
921
+ }
922
+
923
+ public static function getRemoteIp() {
924
+ if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {
925
+ return $_SERVER['HTTP_X_FORWARDED_FOR'];
926
+ }
927
+ if (isset($_SERVER['REMOTE_ADDR'])) {
928
+ return $_SERVER['REMOTE_ADDR'];
929
+ }
930
+ return '';
931
+ }
932
+
933
+ public static function getRemoteHost(){
934
+ return @gethostbyaddr(self::getRemoteIp());
935
+ }
936
+ }
937
+
938
+ } //end La_Http
939
+
940
+ if (!interface_exists('La_Templates_HasAttributes', false)) {
941
+ interface La_Templates_HasAttributes {
942
+ function getAttributes();
943
+ }
944
+
945
+ } //end La_Templates_HasAttributes
946
+
947
+ if (!class_exists('La_Data_RecordHeader', false)) {
948
+ class La_Data_RecordHeader extends La_Object {
949
+ private $ids = array();
950
+
951
+ /**
952
+ * Create Record header object
953
+ *
954
+ * @param array $headerArray
955
+ */
956
+ public function __construct($headerArray = null) {
957
+ if($headerArray === null) {
958
+ return;
959
+ }
960
+
961
+ foreach ($headerArray as $id) {
962
+ $this->add($id);
963
+ }
964
+ }
965
+
966
+ public function contains($id) {
967
+ return array_key_exists($id, $this->ids);
968
+ }
969
+
970
+ public function add($id) {
971
+ if($this->contains($id)) {
972
+ return;
973
+ }
974
+
975
+ $this->ids[$id] = count($this->ids);
976
+ }
977
+
978
+ public function getIds() {
979
+ return array_keys($this->ids);
980
+ }
981
+
982
+ public function getIndex($id) {
983
+ if(!$this->contains($id)) {
984
+ throw new La_Exception("Unknown column '" . $id ."'");
985
+ }
986
+ return $this->ids[$id];
987
+ }
988
+
989
+ public function getSize() {
990
+ return count($this->ids);
991
+ }
992
+
993
+ public function toArray() {
994
+ $response = array();
995
+ foreach ($this->ids as $columnId => $columnIndex) {
996
+ $response[] = $columnId;
997
+ }
998
+ return $response;
999
+ }
1000
+
1001
+ public function toObject() {
1002
+ $result = array();
1003
+ foreach ($this->ids as $columnId => $columnIndex) {
1004
+ $result[] = $columnId;
1005
+ }
1006
+ return $result;
1007
+ }
1008
+ }
1009
+
1010
+
1011
+ } //end La_Data_RecordHeader
1012
+
1013
+ if (!interface_exists('La_Data_Row', false)) {
1014
+ interface La_Data_Row {
1015
+ public function get($name);
1016
+
1017
+ public function set($name, $value);
1018
+ }
1019
+
1020
+ } //end La_Data_Row
1021
+
1022
+ if (!class_exists('La_Data_Record', false)) {
1023
+ class La_Data_Record extends La_Object implements Iterator, La_Rpc_Serializable,
1024
+ La_Templates_HasAttributes, La_Data_Row {
1025
+ private $record;
1026
+ /**
1027
+ *
1028
+ * @var La_Data_RecordHeader
1029
+ */
1030
+ private $header;
1031
+ private $position;
1032
+
1033
+ /**
1034
+ * Create record
1035
+ *
1036
+ * @param array $header
1037
+ * @param array $array values of record from array
1038
+ */
1039
+ public function __construct($header, $array = array()) {
1040
+ if (is_array($header)) {
1041
+ $header = new La_Data_RecordHeader($header);
1042
+ }
1043
+ $this->header = $header;
1044
+ $this->record = array_values($array);
1045
+ while(count($this->record) < $this->header->getSize()) {
1046
+ $this->record[] = null;
1047
+ }
1048
+ }
1049
+
1050
+ function getAttributes() {
1051
+ $ret = array();
1052
+ foreach ($this as $name => $value) {
1053
+ $ret[$name] = $value;
1054
+ }
1055
+ return $ret;
1056
+ }
1057
+
1058
+ public function contains($id) {
1059
+ return $this->header->contains($id);
1060
+ }
1061
+
1062
+ public function get($id) {
1063
+ $index = $this->header->getIndex($id);
1064
+ return $this->record[$index];
1065
+ }
1066
+
1067
+ public function set($id, $value) {
1068
+ $index = $this->header->getIndex($id);
1069
+ $this->record[$index] = $value;
1070
+ }
1071
+
1072
+ public function add($id, $value) {
1073
+ $this->header->add($id);
1074
+ $this->set($id, $value);
1075
+ }
1076
+
1077
+ public function toObject() {
1078
+ return $this->record;
1079
+ }
1080
+
1081
+ public function loadFromObject(array $array) {
1082
+ $this->record = $array;
1083
+ }
1084
+
1085
+ public function toText() {
1086
+ return implode('-', $this->record);
1087
+ }
1088
+
1089
+ public function current() {
1090
+ if(!isset($this->record[$this->position])) {
1091
+ return null;
1092
+ }
1093
+ return $this->record[$this->position];
1094
+ }
1095
+
1096
+ public function key() {
1097
+ $ids = $this->header->getIds();
1098
+ return $ids[$this->position];
1099
+ }
1100
+
1101
+ public function next() {
1102
+ $this->position++;
1103
+ }
1104
+
1105
+ public function rewind() {
1106
+ $this->position = 0;
1107
+ }
1108
+
1109
+ public function valid() {
1110
+ return $this->position < $this->header->getSize();
1111
+ }
1112
+ }
1113
+
1114
+
1115
+ } //end La_Data_Record
1116
+
1117
+ if (!class_exists('La_Data_Grid', false)) {
1118
+ class La_Data_Grid extends La_Object {
1119
+ /**
1120
+ * @var La_Data_RecordSet
1121
+ */
1122
+ private $recordset;
1123
+ private $totalCount;
1124
+
1125
+ public function loadFromObject(stdClass $object) {
1126
+ $this->recordset = new La_Data_RecordSet();
1127
+ $this->recordset->loadFromObject($object->R);
1128
+ $this->totalCount = $object->C;
1129
+ }
1130
+
1131
+ /**
1132
+ * @return La_Data_RecordSet
1133
+ */
1134
+ public function getRecordset() {
1135
+ return $this->recordset;
1136
+ }
1137
+
1138
+ public function getTotalCount() {
1139
+ return $this->totalCount;
1140
+ }
1141
+ }
1142
+
1143
+
1144
+ } //end La_Data_Grid
1145
+
1146
+ if (!class_exists('La_Data_Filter', false)) {
1147
+ class La_Data_Filter extends La_Object implements La_Rpc_Serializable {
1148
+ const LIKE = "L";
1149
+ const NOT_LIKE = "NL";
1150
+ const EQUALS = "E";
1151
+ const NOT_EQUALS = "NE";
1152
+
1153
+ const DATE_EQUALS = "D=";
1154
+ const DATE_GREATER = "D>";
1155
+ const DATE_LOWER = "D<";
1156
+ const DATE_EQUALS_GREATER = "D>=";
1157
+ const DATE_EQUALS_LOWER = "D<=";
1158
+ const DATERANGE_IS = "DP";
1159
+ const TIME_EQUALS = "T=";
1160
+ const TIME_GREATER = "T>";
1161
+ const TIME_LOWER = "T<";
1162
+ const TIME_EQUALS_GREATER = "T>=";
1163
+ const TIME_EQUALS_LOWER = "T<=";
1164
+
1165
+ const RANGE_TODAY = 'T';
1166
+ const RANGE_YESTERDAY = 'Y';
1167
+ const RANGE_LAST_7_DAYS = 'L7D';
1168
+ const RANGE_LAST_30_DAYS = 'L30D';
1169
+ const RANGE_LAST_90_DAYS = 'L90D';
1170
+ const RANGE_THIS_WEEK = 'TW';
1171
+ const RANGE_LAST_WEEK = 'LW';
1172
+ const RANGE_LAST_2WEEKS = 'L2W';
1173
+ const RANGE_LAST_WORKING_WEEK = 'LWW';
1174
+ const RANGE_THIS_MONTH = 'TM';
1175
+ const RANGE_LAST_MONTH = 'LM';
1176
+ const RANGE_THIS_YEAR = 'TY';
1177
+ const RANGE_LAST_YEAR = 'LY';
1178
+
1179
+ private $code;
1180
+ private $operator;
1181
+ private $value;
1182
+
1183
+ public function __construct($code, $operator, $value) {
1184
+ $this->code = $code;
1185
+ $this->operator = $operator;
1186
+ $this->value = $value;
1187
+ }
1188
+
1189
+ public function toObject() {
1190
+ return array($this->code, $this->operator, $this->value);
1191
+ }
1192
+
1193
+ public function toText() {
1194
+ throw new La_Exception("Unsupported");
1195
+ }
1196
+ }
1197
+
1198
+
1199
+ } //end La_Data_Filter
1200
+
1201
+ if (!class_exists('La_Rpc_GridRequest', false)) {
1202
+ class La_Rpc_GridRequest extends La_Rpc_Request {
1203
+
1204
+ private $filters = array();
1205
+
1206
+ private $limit = '';
1207
+ private $offset = '';
1208
+
1209
+ private $sortColumn = '';
1210
+ private $sortAscending = false;
1211
+
1212
+ /**
1213
+ * @return La_Data_Grid
1214
+ */
1215
+ public function getGrid() {
1216
+ $response = new La_Data_Grid();
1217
+ $response->loadFromObject($this->getStdResponse());
1218
+ return $response;
1219
+ }
1220
+
1221
+ public function getFilters() {
1222
+ return $this->filters;
1223
+ }
1224
+
1225
+ /**
1226
+ * adds filter to grid
1227
+ *
1228
+ * @param unknown_type $code
1229
+ * @param unknown_type $operator
1230
+ * @param unknown_type $value
1231
+ */
1232
+ public function addFilter($code, $operator, $value) {
1233
+ $this->filters[] = new La_Data_Filter($code, $operator, $value);
1234
+ }
1235
+
1236
+ public function setLimit($offset, $limit) {
1237
+ $this->offset = $offset;
1238
+ $this->limit = $limit;
1239
+ }
1240
+
1241
+ public function setSorting($sortColumn, $sortAscending = false) {
1242
+ $this->sortColumn = $sortColumn;
1243
+ $this->sortAscending = $sortAscending;
1244
+ }
1245
+
1246
+ public function send() {
1247
+ if(count($this->filters) > 0) {
1248
+ $this->addParam("filters", $this->addFiltersParameter());
1249
+ }
1250
+ if($this->sortColumn !== '') {
1251
+ $this->addParam("sort_col", $this->sortColumn);
1252
+ $this->addParam("sort_asc", ($this->sortAscending ? 'true' : 'false'));
1253
+ }
1254
+ if($this->offset !== '') {
1255
+ $this->addParam("offset", $this->offset);
1256
+ }
1257
+ if($this->limit !== '') {
1258
+ $this->addParam("limit", $this->limit);
1259
+ }
1260
+
1261
+ parent::send();
1262
+ }
1263
+
1264
+ private function addFiltersParameter() {
1265
+ $filters = new La_Rpc_Array();
1266
+
1267
+ foreach($this->filters as $filter) {
1268
+ $filters->add($filter);
1269
+ }
1270
+
1271
+ return $filters;
1272
+ }
1273
+ }
1274
+
1275
+
1276
+
1277
+ } //end La_Rpc_GridRequest
1278
+
1279
+ if (!class_exists('La_Data_RecordSet', false)) {
1280
+ class La_Data_RecordSet extends La_Object implements IteratorAggregate, La_Rpc_Serializable {
1281
+
1282
+ const SORT_ASC = 'ASC';
1283
+ const SORT_DESC = 'DESC';
1284
+
1285
+ protected $_array;
1286
+ /**
1287
+ * @var La_Data_RecordHeader
1288
+ */
1289
+ private $_header;
1290
+
1291
+ function __construct() {
1292
+ $this->init();
1293
+ }
1294
+
1295
+ public function loadFromArray($rows) {
1296
+ $this->setHeader($rows[0]);
1297
+
1298
+ for ($i = 1; $i < count($rows); $i++) {
1299
+ $this->add($rows[$i]);
1300
+ }
1301
+ }
1302
+
1303
+ public function setHeader($header) {
1304
+ if($header instanceof La_Data_RecordHeader) {
1305
+ $this->_header = $header;
1306
+ return;
1307
+ }
1308
+ $this->_header = new La_Data_RecordHeader($header);
1309
+ }
1310
+
1311
+ /**
1312
+ * @return La_Data_RecordHeader
1313
+ */
1314
+ public function getHeader() {
1315
+ return $this->_header;
1316
+ }
1317
+
1318
+ public function addRecordAtStart(La_Data_Record $record) {
1319
+ array_unshift($this->_array, $record);
1320
+ }
1321
+
1322
+ public function addRecord(La_Data_Record $record) {
1323
+ $this->_array[] = $record;
1324
+ }
1325
+
1326
+ /**
1327
+ * Adds new row to RecordSet
1328
+ *
1329
+ * @param array $record array of data for all columns in record
1330
+ */
1331
+ public function add($record) {
1332
+ $this->addRecord($this->getRecordObject($record));
1333
+ }
1334
+
1335
+ /**
1336
+ * @return La_Data_Record
1337
+ */
1338
+ public function createRecord() {
1339
+ return new La_Data_Record($this->_header);
1340
+ }
1341
+
1342
+ public function toObject() {
1343
+ $response = array();
1344
+ $response[] = $this->_header->toObject();
1345
+ foreach ($this->_array as $record) {
1346
+ $response[] = $record->toObject();
1347
+ }
1348
+ return $response;
1349
+ }
1350
+
1351
+ public function loadFromObject(array $array) {
1352
+ $this->_header = new La_Data_RecordHeader($array[0]);
1353
+ for($i = 1; $i < count($array);$i++) {
1354
+ $record = new La_Data_Record($this->_header);
1355
+ $record->loadFromObject($array[$i]);
1356
+ $this->loadRecordFromObject($record);
1357
+ }
1358
+ }
1359
+
1360
+ public function sort($column, $sortType = 'ASC') {
1361
+ if (!$this->_header->contains($column)) {
1362
+ throw new La_Exception('Undefined column');
1363
+ }
1364
+ $sorter = new La_Data_RecordSet_Sorter($column, $sortType);
1365
+ $this->_array = $sorter->sort($this->_array);
1366
+ }
1367
+
1368
+ protected function loadRecordFromObject(La_Data_Record $record) {
1369
+ $this->_array[] = $record;
1370
+ }
1371
+
1372
+ public function toArray() {
1373
+ $response = array();
1374
+ foreach ($this->_array as $record) {
1375
+ $response[] = $record->getAttributes();
1376
+ }
1377
+ return $response;
1378
+ }
1379
+
1380
+ public function toText() {
1381
+ $text = '';
1382
+ foreach ($this->_array as $record) {
1383
+ $text .= $record->toText() . "<br>\n";
1384
+ }
1385
+ return $text;
1386
+ }
1387
+
1388
+ /**
1389
+ * Return number of rows in recordset
1390
+ *
1391
+ * @return integer
1392
+ */
1393
+ public function getSize() {
1394
+ return count($this->_array);
1395
+ }
1396
+
1397
+ /**
1398
+ * @return La_Data_Record
1399
+ */
1400
+ public function get($i) {
1401
+ return $this->_array[$i];
1402
+ }
1403
+
1404
+ /**
1405
+ * @param array/La_Data_Record $record
1406
+ * @return La_Data_Record
1407
+ */
1408
+ private function getRecordObject($record) {
1409
+ if(!($record instanceof La_Data_Record)) {
1410
+ $record = new La_Data_Record($this->_header->toArray(), $record);
1411
+ }
1412
+ return $record;
1413
+ }
1414
+
1415
+ private function init() {
1416
+ $this->_array = array();
1417
+ $this->_header = new La_Data_RecordHeader();
1418
+ }
1419
+
1420
+ public function clear() {
1421
+ $this->init();
1422
+ }
1423
+
1424
+ public function load(La_SqlBuilder_SelectBuilder $select) {
1425
+ }
1426
+
1427
+ /**
1428
+ *
1429
+ * @return ArrayIterator
1430
+ */
1431
+ public function getIterator() {
1432
+ return new ArrayIterator($this->_array);
1433
+ }
1434
+
1435
+ public function getRecord($keyValue = null) {
1436
+ if(!array_key_exists($keyValue, $this->_array)) {
1437
+ return $this->createRecord();
1438
+ }
1439
+ return $this->_array[$keyValue];
1440
+ }
1441
+
1442
+ public function addColumn($id, $defaultValue = "") {
1443
+ $this->_header->add($id);
1444
+ foreach ($this->_array as $record) {
1445
+ $record->add($id, $defaultValue);
1446
+ }
1447
+ }
1448
+
1449
+ /**
1450
+ * Creates shalow copy of recordset containing only headers
1451
+ *
1452
+ * @return La_Data_RecordSet
1453
+ */
1454
+ public function toShalowRecordSet() {
1455
+ $copy = new La_Data_RecordSet();
1456
+ $copy->setHeader($this->_header->toArray());
1457
+ return $copy;
1458
+ }
1459
+ }
1460
+
1461
+ class La_Data_RecordSet_Sorter {
1462
+
1463
+ private $sortColumn;
1464
+ private $sortType;
1465
+
1466
+ function __construct($column, $sortType) {
1467
+ $this->sortColumn = $column;
1468
+ $this->sortType = $sortType;
1469
+ }
1470
+
1471
+ public function sort(array $sortedArray) {
1472
+ usort($sortedArray, array($this, 'compareRecords'));
1473
+ return $sortedArray;
1474
+ }
1475
+
1476
+ private function compareRecords($record1, $record2) {
1477
+ if ($record1->get($this->sortColumn) == $record2->get($this->sortColumn)) {
1478
+ return 0;
1479
+ }
1480
+ return $this->compare($record1->get($this->sortColumn), $record2->get($this->sortColumn));
1481
+ }
1482
+
1483
+ private function compare($value1, $value2) {
1484
+ if ($this->sortType == La_Data_RecordSet::SORT_ASC) {
1485
+ return ($value1 < $value2) ? -1 : 1;
1486
+ }
1487
+ return ($value1 < $value2) ? 1 : -1;
1488
+ }
1489
+ }
1490
+
1491
+ } //end La_Data_RecordSet
1492
+
1493
+ if (!class_exists('La_Data_IndexedRecordSet', false)) {
1494
+ class La_Data_IndexedRecordSet extends La_Data_RecordSet {
1495
+ private $key;
1496
+
1497
+ /**
1498
+ *
1499
+ * @param int $keyIndex specifies which column should be used as a key
1500
+ */
1501
+ function __construct($key) {
1502
+ parent::__construct();
1503
+ $this->key = $key;
1504
+ }
1505
+
1506
+ public function addRecord(La_Data_Record $record) {
1507
+ $this->_array[$record->get($this->key)] = $record;
1508
+ }
1509
+
1510
+ /**
1511
+ * @param String $keyValue
1512
+ * @return La_Data_Record
1513
+ */
1514
+ public function createRecord($keyValue = null) {
1515
+ if($keyValue === null) {
1516
+ return parent::createRecord();
1517
+ }
1518
+ if(!array_key_exists($keyValue, $this->_array)) {
1519
+ $record = $this->createRecord();
1520
+ $record->set($this->key, $keyValue);
1521
+ $this->addRecord($record);
1522
+ }
1523
+ return $this->_array[$keyValue];
1524
+ }
1525
+
1526
+ protected function loadRecordFromObject(La_Data_Record $record) {
1527
+ $this->_array[$record->get($this->key)] = $record;
1528
+ }
1529
+
1530
+ /**
1531
+ * @param String $keyValue
1532
+ * @return La_Data_Record
1533
+ */
1534
+ public function getRecord($keyValue = null) {
1535
+ if (!isset($this->_array[$keyValue])) {
1536
+ throw new La_Data_RecordSetNoRowException($keyValue);
1537
+ }
1538
+ return $this->_array[$keyValue];
1539
+ }
1540
+
1541
+ /**
1542
+ * @param String $keyValue
1543
+ * @return boolean
1544
+ */
1545
+ public function existsRecord($keyValue) {
1546
+ return isset($this->_array[$keyValue]);
1547
+ }
1548
+
1549
+ /**
1550
+ * @param String $sortOptions (SORT_ASC, SORT_DESC, SORT_REGULAR, SORT_NUMERIC, SORT_STRING)
1551
+ * @return boolean
1552
+ */
1553
+ public function sortByKeyValue($sortOptions) {
1554
+ return array_multisort($this->_array, $sortOptions);
1555
+ }
1556
+ }
1557
+
1558
+
1559
+ } //end La_Data_IndexedRecordSet
1560
+
1561
+ if (!class_exists('La_Net_Http_Request', false)) {
1562
+ class La_Net_Http_Request extends La_Object {
1563
+ const CRLF = "\r\n";
1564
+
1565
+ private $method = 'GET';
1566
+ private $url;
1567
+
1568
+ //proxy server
1569
+ private $proxyServer = '';
1570
+ private $proxyPort = '';
1571
+ private $proxyUser = '';
1572
+ private $proxyPassword = '';
1573
+
1574
+ //URL components
1575
+ private $scheme = 'http';
1576
+ private $host = '';
1577
+ private $port = 80;
1578
+ private $http_user = '';
1579
+ private $http_password = '';
1580
+ private $path = '';
1581
+ private $query = '';
1582
+ private $fragment = '';
1583
+ private $cookies = '';
1584
+
1585
+ private $body = '';
1586
+ private $headers = array();
1587
+
1588
+ public function setCookies($cookies) {
1589
+ $this->cookies = $cookies;
1590
+ }
1591
+
1592
+ public function getCookies() {
1593
+ return $this->cookies;
1594
+ }
1595
+
1596
+ public function getCookiesHeader() {
1597
+ return "Cookie: " . $this->cookies;
1598
+ }
1599
+
1600
+ public function setUrl($url) {
1601
+ $this->url = $url;
1602
+ $this->parseUrl();
1603
+ }
1604
+
1605
+ public function getUrl() {
1606
+ return $this->url;
1607
+ }
1608
+
1609
+ private function parseUrl() {
1610
+ $components = @parse_url($this->url);
1611
+ if (!$components) {
1612
+ return;
1613
+ }
1614
+ if (array_key_exists('scheme', $components)) {
1615
+ $this->scheme = $components['scheme'];
1616
+ }
1617
+ if (array_key_exists('host', $components)) {
1618
+ $this->host = $components['host'];
1619
+ }
1620
+ if (array_key_exists('port', $components)) {
1621
+ $this->port = $components['port'];
1622
+ }
1623
+ if (array_key_exists('user', $components)) {
1624
+ $this->http_user = $components['user'];
1625
+ }
1626
+ if (array_key_exists('pass', $components)) {
1627
+ $this->http_password = $components['pass'];
1628
+ }
1629
+ if (array_key_exists('path', $components)) {
1630
+ $this->path = $components['path'];
1631
+ }
1632
+ if (array_key_exists('query', $components)) {
1633
+ $this->query = $components['query'];
1634
+ }
1635
+ if (array_key_exists('fragment', $components)) {
1636
+ $this->fragement = $components['fragment'];
1637
+ }
1638
+ }
1639
+
1640
+ public function getScheme() {
1641
+ return $this->scheme;
1642
+ }
1643
+
1644
+ public function getHost() {
1645
+ if (strlen($this->proxyServer)) {
1646
+ return $this->proxyServer;
1647
+ }
1648
+ return $this->host;
1649
+ }
1650
+
1651
+ public function getPort() {
1652
+ if (strlen($this->proxyServer)) {
1653
+ return $this->proxyPort;
1654
+ }
1655
+
1656
+ if (strlen($this->port)) {
1657
+ return $this->port;
1658
+ }
1659
+ return 80;
1660
+ }
1661
+
1662
+ public function getHttpUser() {
1663
+ return $this->http_user;
1664
+ }
1665
+
1666
+ public function setHttpUser($user) {
1667
+ $this->http_user = $user;
1668
+ }
1669
+
1670
+ public function getHttpPassword() {
1671
+ return $this->http_password;
1672
+ }
1673
+
1674
+ public function setHttpPassword($pass) {
1675
+ $this->http_password = $pass;
1676
+ }
1677
+
1678
+ public function getPath() {
1679
+ return $this->path;
1680
+ }
1681
+
1682
+ public function getQuery() {
1683
+ return $this->query;
1684
+ }
1685
+
1686
+ public function addQueryParam($name, $value) {
1687
+ if (is_array($value)) {
1688
+ foreach($value as $key => $subValue) {
1689
+ $this->addQueryParam($name."[".$key."]", $subValue);
1690
+ }
1691
+ return;
1692
+ }
1693
+ $this->query .= ($this->query == '') ? '?' : '&';
1694
+ $this->query .= $name.'='.urlencode($value);
1695
+ }
1696
+
1697
+ public function getFragemnt() {
1698
+ return $this->fragment;
1699
+ }
1700
+
1701
+ /**
1702
+ * Set if request method is GET or POST
1703
+ *
1704
+ * @param string $method possible values are POST or GET
1705
+ */
1706
+ public function setMethod($method) {
1707
+ $method = strtoupper($method);
1708
+ if ($method != 'GET' && $method != 'POST') {
1709
+ throw new La_Exception('Unsupported HTTP method: ' . $method);
1710
+ }
1711
+ $this->method = $method;
1712
+ }
1713
+
1714
+ /**
1715
+ * get the request method
1716
+ *
1717
+ * @access public
1718
+ * @return string
1719
+ */
1720
+ public function getMethod() {
1721
+ return $this->method;
1722
+ }
1723
+
1724
+ /**
1725
+ * In case request should be redirected through proxy server, set proxy server settings
1726
+ * This function should be called after function setHost !!!
1727
+ *
1728
+ * @param string $server
1729
+ * @param string $port
1730
+ * @param string $user
1731
+ * @param string $password
1732
+ */
1733
+ public function setProxyServer($server, $port, $user, $password) {
1734
+ $this->proxyServer = $server;
1735
+ $this->proxyPort = $port;
1736
+ $this->proxyUser = $user;
1737
+ $this->proxyPassword = $password;
1738
+ }
1739
+
1740
+ public function getProxyServer() {
1741
+ return $this->proxyServer;
1742
+ }
1743
+
1744
+ public function getProxyPort() {
1745
+ return $this->proxyPort;
1746
+ }
1747
+
1748
+ public function getProxyUser() {
1749
+ return $this->proxyUser;
1750
+ }
1751
+
1752
+ public function getProxyPassword() {
1753
+ return $this->proxyPassword;
1754
+ }
1755
+
1756
+ public function setBody($body) {
1757
+ $this->body = $body;
1758
+ }
1759
+
1760
+ public function getBody() {
1761
+ return $this->body;
1762
+ }
1763
+
1764
+ /**
1765
+ * Set header value
1766
+ *
1767
+ * @param string $name
1768
+ * @param string $value
1769
+ */
1770
+ public function setHeader($name, $value) {
1771
+ $this->headers[$name] = $value;
1772
+ }
1773
+
1774
+ /**
1775
+ * Get header value
1776
+ *
1777
+ * @param string $name
1778
+ * @return string
1779
+ */
1780
+ public function getHeader($name) {
1781
+ if (array_key_exists($name, $this->headers)) {
1782
+ return $this->headers[$name];
1783
+ }
1784
+ return null;
1785
+ }
1786
+
1787
+ /**
1788
+ * Return array of headers
1789
+ *
1790
+ * @return array
1791
+ */
1792
+ public function getHeaders() {
1793
+ $headers = array();
1794
+ foreach ($this->headers as $headerName => $headerValue) {
1795
+ $headers[] = "$headerName: $headerValue";
1796
+ }
1797
+ return $headers;
1798
+ }
1799
+
1800
+ private function initHeaders() {
1801
+ if ($this->getPort() == '80') {
1802
+ $this->setHeader('Host', $this->getHost());
1803
+ } else {
1804
+ $this->setHeader('Host', $this->getHost() . ':' . $this->getPort());
1805
+ }
1806
+ if (isset($_SERVER['HTTP_USER_AGENT'])) {
1807
+ $this->setHeader('User-Agent', $_SERVER['HTTP_USER_AGENT']);
1808
+ }
1809
+ if (isset($_SERVER['HTTP_ACCEPT'])) {
1810
+ $this->setHeader('Accept', $_SERVER['HTTP_ACCEPT']);
1811
+ }
1812
+ if (isset($_SERVER['HTTP_ACCEPT_CHARSET'])) {
1813
+ $this->setHeader('Accept-Charset', $_SERVER['HTTP_ACCEPT_CHARSET']);
1814
+ }
1815
+ if (isset($_SERVER['HTTP_ACCEPT_LANGUAGE'])) {
1816
+ $this->setHeader('Accept-Language', $_SERVER['HTTP_ACCEPT_LANGUAGE']);
1817
+ }
1818
+ if (isset($_SERVER['HTTP_REFERER'])) {
1819
+ $this->setHeader('Referer', $_SERVER['HTTP_REFERER']);
1820
+ }
1821
+ if ($this->getMethod() == 'POST' && !strlen($this->getHeader("Content-Type"))) {
1822
+ $this->setHeader("Content-Type", "application/x-www-form-urlencoded");
1823
+ }
1824
+
1825
+ $this->setHeader('Content-Length', strlen($this->getBody()));
1826
+ $this->setHeader('Connection', 'close');
1827
+
1828
+ if (strlen($this->proxyUser)) {
1829
+ $this->setHeader('Proxy-Authorization',
1830
+ 'Basic ' . base64_encode ($this->proxyUser . ':' . $this->proxyPassword));
1831
+ }
1832
+
1833
+ }
1834
+
1835
+ public function getUri() {
1836
+ $uri = $this->getPath();
1837
+ if (strlen($this->getQuery())) {
1838
+ $uri .= '?' . $this->getQuery();
1839
+ }
1840
+ return $uri;
1841
+ }
1842
+
1843
+ public function toString() {
1844
+ $this->initHeaders();
1845
+ $out = sprintf('%s %s HTTP/1.0' . self::CRLF, $this->getMethod(), $this->getUri());
1846
+ $out .= implode(self::CRLF, $this->getHeaders()) . self::CRLF . $this->getCookiesHeader() . self::CRLF;
1847
+ $out .= self::CRLF . $this->getBody();
1848
+ return $out;
1849
+ }
1850
+
1851
+ }
1852
+
1853
+ } //end La_Net_Http_Request
1854
+
1855
+ if (!class_exists('La_Net_Http_ClientBase', false)) {
1856
+ abstract class La_Net_Http_ClientBase extends La_Object {
1857
+ const CONNECTION_TIMEOUT = 20;
1858
+
1859
+ //TODO: rename this method to "send()"
1860
+ /**
1861
+ * @param La_Net_Http_Request $request
1862
+ * @return La_Net_Http_Response
1863
+ */
1864
+ public function execute(La_Net_Http_Request $request) {
1865
+
1866
+ if (!$this->isNetworkingEnabled()) {
1867
+ throw new La_Exception($this->_('Network connections are disabled'));
1868
+ }
1869
+
1870
+ if (!strlen($request->getUrl())) {
1871
+ throw new La_Exception('No URL defined.');
1872
+ }
1873
+
1874
+ $this->setProxyServer($request);
1875
+ if (La_Php::isFunctionEnabled('curl_init')) {
1876
+ return $this->executeWithCurl($request);
1877
+ } else {
1878
+ return $this->executeWithSocketOpen($request);
1879
+ }
1880
+ }
1881
+
1882
+ protected abstract function isNetworkingEnabled();
1883
+
1884
+ /**
1885
+ * @param La_Net_Http_Request $request
1886
+ * @return La_Net_Http_Response
1887
+ */
1888
+ private function executeWithSocketOpen(La_Net_Http_Request $request) {
1889
+ $scheme = ($request->getScheme() == 'ssl' || $request->getScheme() == 'https') ? 'ssl://' : '';
1890
+ $proxySocket = @fsockopen($scheme . $request->getHost(), $request->getPort(), $errorNr,
1891
+ $errorMessage, self::CONNECTION_TIMEOUT);
1892
+
1893
+ if($proxySocket === false) {
1894
+ $gpfErrorMessage = $this->_sys('Could not connect to server: %s:%s, Failed with error: %s', $request->getHost(), $request->getPort(), $errorMessage);
1895
+ La_Log::error($gpfErrorMessage);
1896
+ throw new La_Exception($gpfErrorMessage);
1897
+ }
1898
+
1899
+ $requestText = $request->toString();
1900
+
1901
+ $result = @fwrite($proxySocket, $requestText);
1902
+ if($result === false || $result != strlen($requestText)) {
1903
+ @fclose($proxySocket);
1904
+ $gpfErrorMessage = $this->_sys('Could not send request to server %s:%s', $request->getHost(), $request->getPort());
1905
+ La_Log::error($gpfErrorMessage);
1906
+ throw new La_Exception($gpfErrorMessage);
1907
+ }
1908
+
1909
+ $result = '';
1910
+ while (false === @feof($proxySocket)) {
1911
+ try {
1912
+ if(false === ($data = @fread($proxySocket, 8192))) {
1913
+ La_Log::error($this->_sys('Could not read from proxy socket'));
1914
+ throw new La_Exception("could not read from proxy socket");
1915
+ }
1916
+ $result .= $data;
1917
+ } catch (Exception $e) {
1918
+ La_Log::error($this->_sys('Proxy failed: %s', $e->getMessage()));
1919
+ @fclose($proxySocket);
1920
+ throw new La_Exception($this->_('Proxy failed: %s', $e->getMessage()));
1921
+ }
1922
+ }
1923
+ @fclose($proxySocket);
1924
+
1925
+ $response = new La_Net_Http_Response();
1926
+ $response->setResponseText($result);
1927
+
1928
+ return $response;
1929
+ }
1930
+
1931
+
1932
+ /**
1933
+ * @param La_Net_Http_Request $request
1934
+ * @return La_Net_Http_Response
1935
+ * */
1936
+ private function executeWithCurl(La_Net_Http_Request $request) {
1937
+ $session = curl_init($request->getUrl());
1938
+
1939
+ if ($request->getMethod() == 'POST') {
1940
+ @curl_setopt ($session, CURLOPT_POST, true);
1941
+ @curl_setopt ($session, CURLOPT_POSTFIELDS, $request->getBody());
1942
+ }
1943
+
1944
+ $cookies = $request->getCookies();
1945
+ if($cookies) {
1946
+ @curl_setopt($session, CURLOPT_COOKIE, $cookies);
1947
+ }
1948
+
1949
+ @curl_setopt($session, CURLOPT_HEADER, true);
1950
+ @curl_setopt($session, CURLOPT_CONNECTTIMEOUT, self::CONNECTION_TIMEOUT);
1951
+ @curl_setopt($session, CURLOPT_HTTPHEADER, $request->getHeaders());
1952
+ @curl_setopt($session, CURLOPT_FOLLOWLOCATION, true);
1953
+ @curl_setopt($session, CURLOPT_RETURNTRANSFER, true);
1954
+ if ($request->getHttpPassword() != '' && $request->getHttpUser() != '') {
1955
+ @curl_setopt($session, CURLOPT_USERPWD, $request->getHttpUser() . ":" . $request->getHttpPassword());
1956
+ @curl_setopt($ch, CURLOPT_HTTPAUTH, CURLAUTH_BASIC);
1957
+ }
1958
+ @curl_setopt ($session, CURLOPT_SSL_VERIFYHOST, 0);
1959
+ @curl_setopt ($session, CURLOPT_SSL_VERIFYPEER, 0);
1960
+
1961
+ $this->setupCurlProxyServer($session, $request);
1962
+
1963
+ // Make the call
1964
+ $result = curl_exec($session);
1965
+ $error = curl_error($session);
1966
+
1967
+ curl_close($session);
1968
+
1969
+ if (strlen($error)) {
1970
+ throw new La_Exception("Curl error: " . $error);
1971
+ }
1972
+
1973
+ $response = new La_Net_Http_Response();
1974
+ $response->setResponseText($result);
1975
+
1976
+ return $response;
1977
+ }
1978
+
1979
+ protected function setProxyServer(La_Net_Http_Request $request) {
1980
+ $request->setProxyServer('', '', '', '');
1981
+ }
1982
+
1983
+ private function setupCurlProxyServer($curlSession, La_Net_Http_Request $request) {
1984
+ if (strlen($request->getProxyServer()) && strlen($request->getProxyPort())) {
1985
+ @curl_setopt($curlSession, CURLOPT_PROXY, $request->getProxyServer() . ':' . $request->getProxyPort());
1986
+ if (strlen($request->getProxyUser())) {
1987
+ @curl_setopt($curlSession, CURLOPT_PROXYUSERPWD, $request->getProxyUser() . ':' . $request->getProxyPassword());
1988
+ }
1989
+ }
1990
+ }
1991
+ }
1992
+
1993
+ } //end La_Net_Http_ClientBase
1994
+
1995
+ if (!class_exists('La_Net_Http_Response', false)) {
1996
+ class La_Net_Http_Response extends La_Object {
1997
+
1998
+ private $responseText = '';
1999
+ private $header = '';
2000
+ private $body = '';
2001
+
2002
+ public function setResponseText($responseText) {
2003
+ $this->responseText = $responseText;
2004
+ $this->parse();
2005
+ }
2006
+
2007
+ public function getHeadersText() {
2008
+ return $this->header;
2009
+ }
2010
+
2011
+ private function getHeaderPosition($pos) {
2012
+ return strpos($this->responseText, "\r\n\r\nHTTP", $pos);
2013
+ }
2014
+
2015
+ public function getBody() {
2016
+ return $this->body;
2017
+ }
2018
+
2019
+ private function parse() {
2020
+ $offset = 0;
2021
+ while ($this->getHeaderPosition($offset)) {
2022
+ $offset = $this->getHeaderPosition($offset) + 4;
2023
+ }
2024
+ if (($pos = strpos($this->responseText, "\r\n\r\n", $offset)) > 0) {
2025
+ $this->body = substr($this->responseText, $pos + 4);
2026
+ $this->header = substr($this->responseText, $offset, $pos - $offset);
2027
+ return;
2028
+ }
2029
+ $this->body = '';
2030
+ $this->header = '';
2031
+ }
2032
+ }
2033
+
2034
+ } //end La_Net_Http_Response
2035
+
2036
+ if (!class_exists('La_Rpc_Form', false)) {
2037
+ class La_Rpc_Form extends La_Object implements La_Rpc_Serializable, IteratorAggregate {
2038
+ const FIELD_NAME = "name";
2039
+ const FIELD_VALUE = "value";
2040
+ const FIELD_ERROR = "error";
2041
+ const FIELD_VALUES = "values";
2042
+
2043
+ private $isError = false;
2044
+ private $errorMessage = "";
2045
+ private $infoMessage = "";
2046
+ private $status;
2047
+ /**
2048
+ * @var La_Data_IndexedRecordSet
2049
+ */
2050
+ private $fields;
2051
+ /**
2052
+ * @var La_Rpc_Form_Validator_FormValidatorCollection
2053
+ */
2054
+ private $validators;
2055
+
2056
+ public function __construct(La_Rpc_Params $params = null) {
2057
+ $this->fields = new La_Data_IndexedRecordSet(self::FIELD_NAME);
2058
+
2059
+ $header = new La_Data_RecordHeader();
2060
+ $header->add(self::FIELD_NAME);
2061
+ $header->add(self::FIELD_VALUE);
2062
+ $header->add(self::FIELD_VALUES);
2063
+ $header->add(self::FIELD_ERROR);
2064
+ $this->fields->setHeader($header);
2065
+
2066
+ $this->validator = new La_Rpc_Form_Validator_FormValidatorCollection($this);
2067
+
2068
+ if($params) {
2069
+ $this->loadFieldsFromArray($params->get("fields"));
2070
+ }
2071
+ }
2072
+
2073
+ /**
2074
+ * @param $validator
2075
+ * @param $fieldName
2076
+ * @param $fieldLabel
2077
+ */
2078
+ public function addValidator(La_Rpc_Form_Validator_Validator $validator, $fieldName, $fieldLabel = null) {
2079
+ $this->validator->addValidator($validator, $fieldName, $fieldLabel);
2080
+ }
2081
+
2082
+ /**
2083
+ * @return boolean
2084
+ */
2085
+ public function validate() {
2086
+ return $this->validator->validate();
2087
+ }
2088
+
2089
+ public function loadFieldsFromArray($fields) {
2090
+ for ($i = 1; $i < count($fields); $i++) {
2091
+ $field = $fields[$i];
2092
+ $this->fields->add($field);
2093
+ }
2094
+ }
2095
+
2096
+ /**
2097
+ *
2098
+ * @return ArrayIterator
2099
+ */
2100
+ public function getIterator() {
2101
+ return $this->fields->getIterator();
2102
+ }
2103
+
2104
+ public function addField($name, $value) {
2105
+ $record = $this->fields->createRecord($name);
2106
+ $record->set(self::FIELD_VALUE, $value);
2107
+ }
2108
+
2109
+ public function setField($name, $value, $values = null, $error = "") {
2110
+ $record = $this->fields->createRecord($name);
2111
+ $record->set(self::FIELD_VALUE, $value);
2112
+ $record->set(self::FIELD_VALUES, $values);
2113
+ $record->set(self::FIELD_ERROR, $error);
2114
+ }
2115
+
2116
+ public function setFieldError($name, $error) {
2117
+ $this->isError = true;
2118
+ $record = $this->fields->getRecord($name);
2119
+ $record->set(self::FIELD_ERROR, $error);
2120
+ }
2121
+
2122
+ public function getFieldValue($name) {
2123
+ $record = $this->fields->getRecord($name);
2124
+ return $record->get(self::FIELD_VALUE);
2125
+ }
2126
+
2127
+ public function getFieldError($name) {
2128
+ $record = $this->fields->getRecord($name);
2129
+ return $record->get(self::FIELD_ERROR);
2130
+ }
2131
+
2132
+ public function existsField($name) {
2133
+ return $this->fields->existsRecord($name);
2134
+ }
2135
+
2136
+ public function load(La_Data_Row $row) {
2137
+ foreach($row as $columnName => $columnValue) {
2138
+ $this->setField($columnName, $row->get($columnName));
2139
+ }
2140
+ }
2141
+
2142
+ /**
2143
+ * @return La_Data_IndexedRecordSet
2144
+ */
2145
+ public function getFields() {
2146
+ return $this->fields;
2147
+ }
2148
+
2149
+ public function fill(La_Data_Row $row) {
2150
+ foreach ($this->fields as $field) {
2151
+ try {
2152
+ $row->set($field->get(self::FIELD_NAME), $field->get(self::FIELD_VALUE));
2153
+ } catch (Exception $e) {
2154
+ }
2155
+ }
2156
+ }
2157
+
2158
+ public function toObject() {
2159
+ $response = new stdClass();
2160
+ $response->F = $this->fields->toObject();
2161
+ if ($this->isSuccessful()) {
2162
+ $response->S = Gpf::YES;
2163
+ $response->M = $this->infoMessage;
2164
+ } else {
2165
+ $response->S = Gpf::NO;
2166
+ $response->M = $this->errorMessage;
2167
+ }
2168
+ if (!strlen($response->M)) {
2169
+ unset($response->M);
2170
+ }
2171
+ return $response;
2172
+ }
2173
+
2174
+ public function loadFromObject(stdClass $object) {
2175
+ if ($object->success == Gpf::YES) {
2176
+ $this->setInfoMessage($object->message);
2177
+ } else {
2178
+ $this->setErrorMessage($object->message);
2179
+ }
2180
+
2181
+ $this->fields = new La_Data_IndexedRecordSet(self::FIELD_NAME);
2182
+ $this->fields->loadFromObject($object->fields);
2183
+ }
2184
+
2185
+ public function toText() {
2186
+ return var_dump($this->toObject());
2187
+ }
2188
+
2189
+ public function setErrorMessage($message) {
2190
+ $this->isError = true;
2191
+ $this->errorMessage = $message;
2192
+ }
2193
+
2194
+ public function getErrorMessage() {
2195
+ if ($this->isError) {
2196
+ return $this->errorMessage;
2197
+ }
2198
+ return "";
2199
+ }
2200
+
2201
+ public function setInfoMessage($message) {
2202
+ $this->infoMessage = $message;
2203
+ }
2204
+
2205
+ public function setSuccessful() {
2206
+ $this->isError = false;
2207
+ }
2208
+
2209
+ public function getInfoMessage() {
2210
+ if ($this->isError) {
2211
+ return "";
2212
+ }
2213
+ return $this->infoMessage;
2214
+ }
2215
+
2216
+
2217
+ /**
2218
+ * @return boolean
2219
+ */
2220
+ public function isSuccessful() {
2221
+ return !$this->isError;
2222
+ }
2223
+
2224
+ /**
2225
+ * @return boolean
2226
+ */
2227
+ public function isError() {
2228
+ return $this->isError;
2229
+ }
2230
+
2231
+ public function getDefaultErrorMessage() {
2232
+ return $this->_('There were errors, please check highlighted fields');
2233
+ }
2234
+ }
2235
+
2236
+
2237
+ } //end La_Rpc_Form
2238
+
2239
+ if (!class_exists('La_Rpc_Form_Validator_FormValidatorCollection', false)) {
2240
+ class La_Rpc_Form_Validator_FormValidatorCollection extends La_Object {
2241
+
2242
+ /**
2243
+ * @var array<La_Rpc_Form_Validator_FieldValidator>
2244
+ */
2245
+ private $validators;
2246
+ /**
2247
+ * @var La_Rpc_Form
2248
+ */
2249
+ private $form;
2250
+
2251
+ public function __construct(La_Rpc_Form $form) {
2252
+ $this->form = $form;
2253
+ $this->validators = array();
2254
+ }
2255
+
2256
+ /**
2257
+ * @param $fieldName
2258
+ * @param $validator
2259
+ */
2260
+ public function addValidator(La_Rpc_Form_Validator_Validator $validator, $fieldName, $fieldLabel = null) {
2261
+ if (!array_key_exists($fieldName, $this->validators)) {
2262
+ $this->validators[$fieldName] = new La_Rpc_Form_Validator_FieldValidator(($fieldLabel === null ? $fieldName : $fieldLabel));
2263
+ }
2264
+ $this->validators[$fieldName]->addValidator($validator);
2265
+ }
2266
+
2267
+ /**
2268
+ * @return boolean
2269
+ */
2270
+ public function validate() {
2271
+ $errorMsg = false;
2272
+ foreach ($this->validators as $fieldName => $fieldValidator) {
2273
+ if (!$fieldValidator->validate($this->form->getFieldValue($fieldName))) {
2274
+ $errorMsg = true;
2275
+ $this->form->setFieldError($fieldName, $fieldValidator->getMessage());
2276
+ }
2277
+ }
2278
+ if ($errorMsg) {
2279
+ $this->form->setErrorMessage($this->form->getDefaultErrorMessage());
2280
+ }
2281
+ return !$errorMsg;
2282
+ }
2283
+ }
2284
+
2285
+ } //end La_Rpc_Form_Validator_FormValidatorCollection
2286
+
2287
+ if (!class_exists('La_Rpc_FormRequest', false)) {
2288
+ class La_Rpc_FormRequest extends La_Rpc_Request {
2289
+ /**
2290
+ * @var La_Rpc_Form
2291
+ */
2292
+ private $fields;
2293
+
2294
+ public function __construct($className, $methodName, La_Api_Session $apiSessionObject = null) {
2295
+ parent::__construct($className, $methodName, $apiSessionObject);
2296
+ $this->fields = new La_Rpc_Form();
2297
+ }
2298
+
2299
+ public function send() {
2300
+ $this->addParam('fields', $this->fields->getFields());
2301
+ parent::send();
2302
+ }
2303
+
2304
+ /**
2305
+ * @return La_Rpc_Form
2306
+ */
2307
+ public function getForm() {
2308
+ $response = new La_Rpc_Form();
2309
+ $response->loadFromObject($this->getStdResponse());
2310
+ return $response;
2311
+ }
2312
+
2313
+ public function setField($name, $value) {
2314
+ if (is_scalar($value) || $value instanceof La_Rpc_Serializable) {
2315
+ $this->fields->setField($name, $value);
2316
+ } else {
2317
+ throw new La_Exception("Not supported value");
2318
+ }
2319
+ }
2320
+
2321
+ public function setFields(La_Data_IndexedRecordSet $fields) {
2322
+ $this->fields->loadFieldsFromArray($fields->toArray());
2323
+ }
2324
+ }
2325
+
2326
+ } //end La_Rpc_FormRequest
2327
+
2328
+ if (!class_exists('La_Rpc_RecordSetRequest', false)) {
2329
+ class La_Rpc_RecordSetRequest extends La_Rpc_Request {
2330
+
2331
+ /**
2332
+ * @return La_Data_IndexedRecordSet
2333
+ */
2334
+ public function getIndexedRecordSet($key) {
2335
+ $response = new La_Data_IndexedRecordSet($key);
2336
+ $response->loadFromObject($this->getStdResponse());
2337
+ return $response;
2338
+ }
2339
+
2340
+
2341
+ /**
2342
+ * @return La_Data_RecordSet
2343
+ */
2344
+ public function getRecordSet() {
2345
+ $response = new La_Data_RecordSet();
2346
+ $response->loadFromObject($this->getStdResponse());
2347
+ return $response;
2348
+ }
2349
+ }
2350
+
2351
+
2352
+ } //end La_Rpc_RecordSetRequest
2353
+
2354
+ if (!class_exists('La_Rpc_DataRequest', false)) {
2355
+ class La_Rpc_DataRequest extends La_Rpc_Request {
2356
+ /**
2357
+ * @var La_Rpc_Data
2358
+ */
2359
+ private $data;
2360
+
2361
+ private $filters = array();
2362
+
2363
+ public function __construct($className, $methodName, La_Api_Session $apiSessionObject = null) {
2364
+ parent::__construct($className, $methodName, $apiSessionObject);
2365
+ $this->data = new La_Rpc_Data();
2366
+ }
2367
+
2368
+ /**
2369
+ * @return La_Rpc_Data
2370
+ */
2371
+ public function getData() {
2372
+ $response = new La_Rpc_Data();
2373
+ $response->loadFromObject($this->getStdResponse());
2374
+ return $response;
2375
+ }
2376
+
2377
+ public function setField($name, $value) {
2378
+ if (is_scalar($value) || $value instanceof La_Rpc_Serializable) {
2379
+ $this->data->setParam($name, $value);
2380
+ } else {
2381
+ throw new La_Exception("Not supported value");
2382
+ }
2383
+ }
2384
+
2385
+ /**
2386
+ * adds filter to grid
2387
+ *
2388
+ * @param unknown_type $code
2389
+ * @param unknown_type $operator
2390
+ * @param unknown_type $value
2391
+ */
2392
+ public function addFilter($code, $operator, $value) {
2393
+ $this->filters[] = new La_Data_Filter($code, $operator, $value);
2394
+ }
2395
+
2396
+ public function send() {
2397
+ $this->addParam('data', $this->data->getParams());
2398
+
2399
+ if(count($this->filters) > 0) {
2400
+ $this->addParam("filters", $this->addFiltersParameter());
2401
+ }
2402
+ parent::send();
2403
+ }
2404
+
2405
+ private function addFiltersParameter() {
2406
+ $filters = new La_Rpc_Array();
2407
+
2408
+ foreach($this->filters as $filter) {
2409
+ $filters->add($filter);
2410
+ }
2411
+
2412
+ return $filters;
2413
+ }
2414
+ }
2415
+
2416
+ } //end La_Rpc_DataRequest
2417
+
2418
+ if (!class_exists('La_Rpc_Data', false)) {
2419
+ class La_Rpc_Data extends La_Object implements La_Rpc_Serializable {
2420
+ const NAME = "name";
2421
+ const VALUE = "value";
2422
+ const DATA = "data";
2423
+ const ID = "id";
2424
+
2425
+ /**
2426
+ * @var La_Data_IndexedRecordSet
2427
+ */
2428
+ private $params;
2429
+
2430
+ /**
2431
+ * @var string
2432
+ */
2433
+ private $id;
2434
+
2435
+
2436
+ /**
2437
+ * @var La_Rpc_FilterCollection
2438
+ */
2439
+ private $filters;
2440
+
2441
+ /**
2442
+ * @var La_Data_IndexedRecordSet
2443
+ */
2444
+ private $response;
2445
+
2446
+ /**
2447
+ *
2448
+ * @return La_Data_IndexedRecordSet
2449
+ */
2450
+ public function getParams() {
2451
+ return $this->params;
2452
+ }
2453
+
2454
+ /**
2455
+ * Create instance to handle DataRequest
2456
+ *
2457
+ * @param La_Rpc_Params $params
2458
+ */
2459
+ public function __construct(La_Rpc_Params $params = null) {
2460
+ if($params === null) {
2461
+ $params = new La_Rpc_Params();
2462
+ }
2463
+
2464
+ $this->filters = new La_Rpc_FilterCollection($params);
2465
+
2466
+ $this->params = new La_Data_IndexedRecordSet(self::NAME);
2467
+ $this->params->setHeader(array(self::NAME, self::VALUE));
2468
+
2469
+ if ($params->exists(self::DATA) !== null) {
2470
+ $this->loadParamsFromArray($params->get(self::DATA));
2471
+ }
2472
+
2473
+ $this->id = $params->get(self::ID);
2474
+
2475
+ $this->response = new La_Data_IndexedRecordSet(self::NAME);
2476
+ $this->response->setHeader(array(self::NAME, self::VALUE));
2477
+ }
2478
+
2479
+ public function addValues(array $values) {
2480
+ foreach ($values as $key => $value) {
2481
+ $this->setValue($key, $value);
2482
+ }
2483
+ }
2484
+
2485
+ /**
2486
+ * Return id
2487
+ *
2488
+ * @return string
2489
+ */
2490
+ public function getId() {
2491
+ return $this->id;
2492
+ }
2493
+
2494
+ /**
2495
+ * Return parameter value
2496
+ *
2497
+ * @param String $name
2498
+ * @return unknown
2499
+ */
2500
+ public function getParam($name) {
2501
+ try {
2502
+ return $this->params->getRecord($name)->get(self::VALUE);
2503
+ } catch (La_Data_RecordSetNoRowException $e) {
2504
+ return null;
2505
+ }
2506
+ }
2507
+
2508
+ public function setParam($name, $value) {
2509
+ self::setValueToRecordset($this->params, $name, $value);
2510
+ }
2511
+
2512
+ public function loadFromObject(array $object) {
2513
+ $this->response->loadFromObject($object);
2514
+ $this->params->loadFromObject($object);
2515
+ }
2516
+
2517
+ /**
2518
+ * @return La_Rpc_FilterCollection
2519
+ */
2520
+ public function getFilters() {
2521
+ return $this->filters;
2522
+ }
2523
+
2524
+ private static function setValueToRecordset(La_Data_IndexedRecordSet $recordset, $name, $value) {
2525
+ try {
2526
+ $record = $recordset->getRecord($name);
2527
+ } catch (La_Data_RecordSetNoRowException $e) {
2528
+ $record = $recordset->createRecord();
2529
+ $record->set(self::NAME, $name);
2530
+ $recordset->addRecord($record);
2531
+ }
2532
+ $record->set(self::VALUE, $value);
2533
+ }
2534
+
2535
+ public function setValue($name, $value) {
2536
+ self::setValueToRecordset($this->response, $name, $value);
2537
+ }
2538
+
2539
+ public function getSize() {
2540
+ return $this->response->getSize();
2541
+ }
2542
+
2543
+ public function getValue($name) {
2544
+ return $this->response->getRecord($name)->get(self::VALUE);
2545
+ }
2546
+
2547
+ public function toObject() {
2548
+ return $this->response->toObject();
2549
+ }
2550
+
2551
+ public function toText() {
2552
+ return $this->response->toText();
2553
+ }
2554
+
2555
+ private function loadParamsFromArray($data) {
2556
+ for ($i = 1; $i < count($data); $i++) {
2557
+ $this->params->add($data[$i]);
2558
+ }
2559
+ }
2560
+ }
2561
+
2562
+ } //end La_Rpc_Data
2563
+
2564
+ if (!class_exists('La_Rpc_FilterCollection', false)) {
2565
+ class La_Rpc_FilterCollection extends La_Object implements IteratorAggregate {
2566
+
2567
+ /**
2568
+ * @var array of La_Filter
2569
+ */
2570
+ private $filters;
2571
+
2572
+ public function __construct(La_Rpc_Params $params = null) {
2573
+ $this->filters = array();
2574
+ if ($params != null) {
2575
+ $this->init($params);
2576
+ }
2577
+ }
2578
+
2579
+ /**
2580
+ * @return La_Rpc_FilterCollection
2581
+ */
2582
+ public static function fromJson($json){
2583
+ $instance = new La_Rpc_FilterCollection();
2584
+ $filters = La_Rpc_Json::decodeStatic($json);
2585
+ foreach ($filters as $filter){
2586
+ $instance->add($filter);
2587
+ }
2588
+ return $instance;
2589
+ }
2590
+
2591
+ public function add(array $filterArray) {
2592
+ $this->filters[] = new La_Filter($filterArray);
2593
+ }
2594
+
2595
+ private function init(La_Rpc_Params $params) {
2596
+ $filtersArray = $params->get("filters");
2597
+ if (!is_array($filtersArray)) {
2598
+ return;
2599
+ }
2600
+ foreach ($filtersArray as $filterArray) {
2601
+ $this->add($filterArray);
2602
+ }
2603
+ }
2604
+
2605
+ /**
2606
+ *
2607
+ * @return ArrayIterator
2608
+ */
2609
+ public function getIterator() {
2610
+ return new ArrayIterator($this->filters);
2611
+ }
2612
+
2613
+ public function addTo(La_SqlBuilder_WhereClause $whereClause) {
2614
+ foreach ($this->filters as $filter) {
2615
+ $filter->addTo($whereClause);
2616
+ }
2617
+ }
2618
+
2619
+ public function addSelectedFilterTo(La_SqlBuilder_WhereClause $whereClause, $filterCode, $columnCode = null) {
2620
+ if ($columnCode == null) {
2621
+ $columnCode = $filterCode;
2622
+ }
2623
+ foreach ($this->filters as $filter) {
2624
+ if ($filter->getCode() == $filterCode) {
2625
+ $oldCode = $filter->getCode();
2626
+ $filter->setCode($columnCode);
2627
+ $filter->addTo($whereClause);
2628
+ $filter->setCode($oldCode);
2629
+ }
2630
+ }
2631
+ }
2632
+
2633
+ /**
2634
+ * Returns first filter with specified code.
2635
+ * If filter with specified code does not exists null is returned.
2636
+ *
2637
+ * @param string $code
2638
+ * @return array<La_Filter>
2639
+ */
2640
+ public function getFilter($code) {
2641
+ $filters = array();
2642
+ foreach ($this->filters as $filter) {
2643
+ if ($filter->getCode() == $code) {
2644
+ $filters[] = $filter;
2645
+ }
2646
+ }
2647
+ return $filters;
2648
+ }
2649
+
2650
+ public function isFilter($code) {
2651
+ foreach ($this->filters as $filter) {
2652
+ if ($filter->getCode() == $code) {
2653
+ return true;
2654
+ }
2655
+ }
2656
+ return false;
2657
+ }
2658
+
2659
+ public function getFilterValue($code) {
2660
+ $filters = $this->getFilter($code);
2661
+ if (count($filters) == 1) {
2662
+ return $filters[0]->getValue();
2663
+ }
2664
+ return "";
2665
+ }
2666
+
2667
+ public function matches(La_Data_Record $row) {
2668
+ foreach ($this->filters as $filter) {
2669
+ if (!$filter->matches($row)) {
2670
+ return false;
2671
+ }
2672
+ }
2673
+ return true;
2674
+ }
2675
+
2676
+ public function getSize() {
2677
+ return count($this->filters);
2678
+ }
2679
+ }
2680
+
2681
+ } //end La_Rpc_FilterCollection
2682
+
2683
+ if (!class_exists('La_Php', false)) {
2684
+ class La_Php {
2685
+
2686
+ /**
2687
+ * Check if function is enabled and exists in php
2688
+ *
2689
+ * @param $functionName
2690
+ * @return boolean Returns true if function exists and is enabled
2691
+ */
2692
+ public static function isFunctionEnabled($functionName) {
2693
+ if (function_exists($functionName) && strstr(ini_get("disable_functions"), $functionName) === false) {
2694
+ return true;
2695
+ }
2696
+ return false;
2697
+ }
2698
+
2699
+ /**
2700
+ * Check if extension is loaded
2701
+ *
2702
+ * @param $extensionName
2703
+ * @return boolean Returns true if extension is loaded
2704
+ */
2705
+ public static function isExtensionLoaded($extensionName) {
2706
+ return extension_loaded($extensionName);
2707
+ }
2708
+
2709
+ }
2710
+
2711
+ } //end La_Php
2712
+
2713
+ if (!class_exists('La_Rpc_ActionRequest', false)) {
2714
+ class La_Rpc_ActionRequest extends La_Rpc_Request {
2715
+
2716
+ /**
2717
+ * @return La_Rpc_Action
2718
+ */
2719
+ public function getAction() {
2720
+ $action = new La_Rpc_Action(new La_Rpc_Params());
2721
+ $action->loadFromObject($this->getStdResponse());
2722
+ return $action;
2723
+ }
2724
+ }
2725
+
2726
+
2727
+ } //end La_Rpc_ActionRequest
2728
+
2729
+ if (!class_exists('La_Rpc_Action', false)) {
2730
+ class La_Rpc_Action extends La_Object implements La_Rpc_Serializable {
2731
+ private $errorMessage = "";
2732
+ private $infoMessage = "";
2733
+ private $successCount = 0;
2734
+ private $errorCount = 0;
2735
+ /**
2736
+ * @var La_Rpc_Params
2737
+ */
2738
+ private $params;
2739
+
2740
+ public function __construct(La_Rpc_Params $params, $infoMessage = '', $errorMessage = '') {
2741
+ $this->params = $params;
2742
+ $this->infoMessage = $infoMessage;
2743
+ $this->errorMessage = $errorMessage;
2744
+ }
2745
+
2746
+ /**
2747
+ * @return Iterator
2748
+ */
2749
+ public function getIds() {
2750
+ $massHandler = new La_Rpc_MassHandler($this->params);
2751
+ return $massHandler->getIds();
2752
+ }
2753
+
2754
+ public function getParam($name) {
2755
+ return $this->params->get($name);
2756
+ }
2757
+
2758
+ public function existsParam($name) {
2759
+ return $this->params->exists($name);
2760
+ }
2761
+
2762
+ /**
2763
+ * Parameter OK is mandatory
2764
+ * Parameter I and E is optional and only if there is value it is sent to client (empty values are not transferred)
2765
+ *
2766
+ * (non-PHPdoc)
2767
+ * @see include/Gpf/Rpc/La_Rpc_Serializable#toObject()
2768
+ */
2769
+ public function toObject() {
2770
+ $response = new stdClass();
2771
+ $response->S = Gpf::YES;
2772
+
2773
+ if ($this->errorCount > 0) {
2774
+ $response->S = Gpf::NO;
2775
+ $response->E = $this->_($this->errorMessage, $this->errorCount);
2776
+ if (!strlen($response->E)) {
2777
+ unset($response->E);
2778
+ }
2779
+ }
2780
+
2781
+ if ($this->successCount > 0) {
2782
+ $response->I = $this->_($this->infoMessage, $this->successCount);
2783
+ if (!strlen($response->I)) {
2784
+ unset($response->I);
2785
+ }
2786
+ }
2787
+
2788
+ return $response;
2789
+ }
2790
+
2791
+ public function loadFromObject(stdClass $object) {
2792
+ $this->errorMessage = $object->errorMessage;
2793
+ $this->infoMessage = $object->infoMessage;
2794
+
2795
+ if($object->success == Gpf::NO) {
2796
+ $this->addError();
2797
+ }
2798
+ }
2799
+
2800
+ public function isError() {
2801
+ return $this->errorCount > 0;
2802
+ }
2803
+
2804
+ public function toText() {
2805
+ if ($this->isError()) {
2806
+ return $this->_($this->errorMessage, $this->errorCount);
2807
+ } else {
2808
+ return $this->_($this->infoMessage, $this->successCount);
2809
+ }
2810
+ }
2811
+
2812
+ public function setErrorMessage($message) {
2813
+ $this->errorMessage = $message;
2814
+ }
2815
+
2816
+ public function getErrorMessage() {
2817
+ return $this->errorMessage;
2818
+ }
2819
+
2820
+ public function setInfoMessage($message) {
2821
+ $this->infoMessage = $message;
2822
+ }
2823
+
2824
+ public function addOk() {
2825
+ $this->successCount++;
2826
+ }
2827
+
2828
+ public function addError() {
2829
+ $this->errorCount++;
2830
+ }
2831
+
2832
+ }
2833
+
2834
+
2835
+ } //end La_Rpc_Action
2836
+
2837
+ if (!class_exists('La_Rpc_Map', false)) {
2838
+ class La_Rpc_Map extends La_Object implements La_Rpc_Serializable {
2839
+
2840
+ function __construct(array $array){
2841
+ $this->array = $array;
2842
+ }
2843
+
2844
+ public function toObject() {
2845
+ return $this->array;
2846
+ }
2847
+
2848
+ public function toText() {
2849
+ return var_dump($this->array);
2850
+ }
2851
+ }
2852
+
2853
+
2854
+ } //end La_Rpc_Map
2855
+
2856
+ if (!class_exists('La_Log', false)) {
2857
+ class La_Log {
2858
+ const CRITICAL = 50;
2859
+ const ERROR = 40;
2860
+ const WARNING = 30;
2861
+ const INFO = 20;
2862
+ const DEBUG = 10;
2863
+
2864
+ /**
2865
+ * @var La_Log_Logger
2866
+ */
2867
+ private static $logger;
2868
+
2869
+ /**
2870
+ * @return La_Log_Logger
2871
+ */
2872
+ private static function getLogger() {
2873
+ if (self::$logger == null) {
2874
+ self::$logger = La_Log_Logger::getInstance();
2875
+ }
2876
+ return self::$logger;
2877
+ }
2878
+
2879
+ private function __construct() {
2880
+ }
2881
+
2882
+ public static function disableType($type) {
2883
+ self::getLogger()->disableType($type);
2884
+ }
2885
+
2886
+ public static function enableAllTypes() {
2887
+ self::getLogger()->enableAllTypes();
2888
+ }
2889
+
2890
+ /**
2891
+ * logs message
2892
+ *
2893
+ * @param string $message
2894
+ * @param string $logLevel
2895
+ * @param string $logGroup
2896
+ */
2897
+ public static function log($message, $logLevel, $logGroup = null) {
2898
+ self::getLogger()->log($message, $logLevel, $logGroup);
2899
+ }
2900
+
2901
+ /**
2902
+ * logs debug message
2903
+ *
2904
+ * @param string $message
2905
+ * @param string $logGroup
2906
+ */
2907
+ public static function debug($message, $logGroup = null) {
2908
+ self::getLogger()->debug($message, $logGroup);
2909
+ }
2910
+
2911
+ /**
2912
+ * logs info message
2913
+ *
2914
+ * @param string $message
2915
+ * @param string $logGroup
2916
+ */
2917
+ public static function info($message, $logGroup = null) {
2918
+ self::getLogger()->info($message, $logGroup);
2919
+ }
2920
+
2921
+ /**
2922
+ * logs warning message
2923
+ *
2924
+ * @param string $message
2925
+ * @param string $logGroup
2926
+ */
2927
+ public static function warning($message, $logGroup = null) {
2928
+ self::getLogger()->warning($message, $logGroup);
2929
+ }
2930
+
2931
+ /**
2932
+ * logs error message
2933
+ *
2934
+ * @param string $message
2935
+ * @param string $logGroup
2936
+ */
2937
+ public static function error($message, $logGroup = null) {
2938
+ self::getLogger()->error($message, $logGroup);
2939
+ }
2940
+
2941
+ /**
2942
+ * logs critical error message
2943
+ *
2944
+ * @param string $message
2945
+ * @param string $logGroup
2946
+ */
2947
+ public static function critical($message, $logGroup = null) {
2948
+ self::getLogger()->critical($message, $logGroup);
2949
+ }
2950
+
2951
+ /**
2952
+ * Attach new log system
2953
+ *
2954
+ * @param string $type
2955
+ * La_Log_LoggerDisplay::TYPE
2956
+ * La_Log_LoggerFile::TYPE
2957
+ * La_Log_LoggerDatabase::TYPE
2958
+ * @param string $logLevel
2959
+ * La_Log::CRITICAL
2960
+ * La_Log::ERROR
2961
+ * La_Log::WARNING
2962
+ * La_Log::INFO
2963
+ * La_Log::DEBUG
2964
+ * @return La_Log_LoggerBase
2965
+ */
2966
+ public static function addLogger($type, $logLevel) {
2967
+ if($type instanceof La_Log_LoggerBase) {
2968
+ return self::getLogger()->addLogger($type, $logLevel);
2969
+ }
2970
+ return self::getLogger()->add($type, $logLevel);
2971
+ }
2972
+
2973
+ public static function removeAll() {
2974
+ self::getLogger()->removeAll();
2975
+ }
2976
+ }
2977
+
2978
+ } //end La_Log
2979
+
2980
+ if (!class_exists('La_Log_Logger', false)) {
2981
+ class La_Log_Logger extends La_Object {
2982
+ /**
2983
+ * @var array
2984
+ */
2985
+ static private $instances = array();
2986
+ /**
2987
+ * @var array
2988
+ */
2989
+ private $loggers = array();
2990
+
2991
+ /**
2992
+ * array of custom parameters
2993
+ */
2994
+ private $customParameters = array();
2995
+
2996
+ private $disabledTypes = array();
2997
+
2998
+ private $group = null;
2999
+ private $type = null;
3000
+ private $logToDisplay = false;
3001
+
3002
+ /**
3003
+ * returns instance of logger class.
3004
+ * You can add instance name, if you want to have multiple independent instances of logger
3005
+ *
3006
+ * @param string $instanceName
3007
+ * @return La_Log_Logger
3008
+ */
3009
+ public static function getInstance($instanceName = '_') {
3010
+ if($instanceName == '') {
3011
+ $instanceName = '_';
3012
+ }
3013
+
3014
+ if (!array_key_exists($instanceName, self::$instances)) {
3015
+ self::$instances[$instanceName] = new La_Log_Logger();
3016
+ }
3017
+ $instance = self::$instances[$instanceName];
3018
+ return $instance;
3019
+ }
3020
+
3021
+ public static function isLoggerInsert($sqlString) {
3022
+ return strpos($sqlString, 'INSERT INTO ' . La_Db_Table_Logs::getName()) !== false;
3023
+ }
3024
+
3025
+ /**
3026
+ * attachs new log system
3027
+ *
3028
+ * @param unknown_type $system
3029
+ * @return La_Log_LoggerBase
3030
+ */
3031
+ public function add($type, $logLevel) {
3032
+ if($type == La_Log_LoggerDisplay::TYPE) {
3033
+ $this->logToDisplay = true;
3034
+ }
3035
+ return $this->addLogger($this->create($type), $logLevel);
3036
+ }
3037
+
3038
+ /**
3039
+ * Checks if logger with te specified type was already initialized
3040
+ *
3041
+ * @param unknown_type $type
3042
+ * @return unknown
3043
+ */
3044
+ public function checkLoggerTypeExists($type) {
3045
+ if(array_key_exists($type, $this->loggers)) {
3046
+ return true;
3047
+ }
3048
+
3049
+ return false;
3050
+ }
3051
+
3052
+ /**
3053
+ * returns true if debugging writes log to display
3054
+ *
3055
+ * @return boolean
3056
+ */
3057
+ public function isLogToDisplay() {
3058
+ return $this->logToDisplay;
3059
+ }
3060
+
3061
+ public function removeAll() {
3062
+ $this->loggers = array();
3063
+ $this->customParameters = array();
3064
+ $this->disabledTypes = array();
3065
+ $this->group = null;
3066
+ }
3067
+
3068
+ /**
3069
+ *
3070
+ * @param La_Log_LoggerBase $logger
3071
+ * @param int $logLevel
3072
+ * @return La_Log_LoggerBase
3073
+ */
3074
+ public function addLogger(La_Log_LoggerBase $logger, $logLevel) {
3075
+ if(!$this->checkLoggerTypeExists($logger->getType())) {
3076
+ $logger->setLogLevel($logLevel);
3077
+ $this->loggers[$logger->getType()] = $logger;
3078
+ return $logger;
3079
+ } else {
3080
+ $ll = new La_Log_LoggerDatabase();
3081
+ $existingLogger = $this->loggers[$logger->getType()];
3082
+ if($existingLogger->getLogLevel() > $logLevel) {
3083
+ $existingLogger->setLogLevel($logLevel);
3084
+ }
3085
+ return $existingLogger;
3086
+ }
3087
+ }
3088
+
3089
+ public function getGroup() {
3090
+ return $this->group;
3091
+ }
3092
+
3093
+ public function setGroup($group = null) {
3094
+ $this->group = $group;
3095
+ if($group === null) {
3096
+ $this->group = La_Common_String::generateId(10);
3097
+ }
3098
+ }
3099
+
3100
+ public function setType($type) {
3101
+ $this->type = $type;
3102
+ }
3103
+
3104
+ /**
3105
+ * function sets custom parameter for the logger
3106
+ *
3107
+ * @param string $name
3108
+ * @param string $value
3109
+ */
3110
+ public function setCustomParameter($name, $value) {
3111
+ $this->customParameters[$name] = $value;
3112
+ }
3113
+
3114
+ /**
3115
+ * returns custom parameter
3116
+ *
3117
+ * @param string $name
3118
+ * @return string
3119
+ */
3120
+ public function getCustomParameter($name) {
3121
+ if(isset($this->customParameters[$name])) {
3122
+ return $this->customParameters[$name];
3123
+ }
3124
+ return '';
3125
+ }
3126
+
3127
+ /**
3128
+ * logs message
3129
+ *
3130
+ * @param string $message
3131
+ * @param string $logLevel
3132
+ * @param string $logGroup
3133
+ */
3134
+ public function log($message, $logLevel, $logGroup = null) {
3135
+ $time = time();
3136
+ $backArr = debug_backtrace();
3137
+ $group = $logGroup;
3138
+ if($this->group !== null) {
3139
+ $group = $this->group;
3140
+ if($logGroup !== null) {
3141
+ $group .= ' ' . $logGroup;
3142
+ }
3143
+ }
3144
+
3145
+ $callingFile = $this->findLogFile();
3146
+ $file = $callingFile['file'];
3147
+ if(isset($callingFile['classVariables'])) {
3148
+ $file .= ' '.$callingFile['classVariables'];
3149
+ }
3150
+ $line = $callingFile['line'];
3151
+
3152
+ $ip = La_Http::getRemoteIp();
3153
+ if ($ip = '') {
3154
+ $ip = '127.0.0.1';
3155
+ }
3156
+
3157
+ foreach ($this->loggers as $logger) {
3158
+ if(!in_array($logger->getType(), $this->disabledTypes)) {
3159
+ $logger->logMessage($time, $message, $logLevel, $group, $ip, $file, $line, $this->type);
3160
+ }
3161
+ }
3162
+ }
3163
+
3164
+ /**
3165
+ * logs debug message
3166
+ *
3167
+ * @param string $message
3168
+ * @param string $logGroup
3169
+ */
3170
+ public function debug($message, $logGroup = null) {
3171
+ $this->log($message, La_Log::DEBUG, $logGroup);
3172
+ }
3173
+
3174
+ /**
3175
+ * logs info message
3176
+ *
3177
+ * @param string $message
3178
+ * @param string $logGroup
3179
+ */
3180
+ public function info($message, $logGroup = null) {
3181
+ $this->log($message, La_Log::INFO, $logGroup);
3182
+ }
3183
+
3184
+ /**
3185
+ * logs warning message
3186
+ *
3187
+ * @param string $message
3188
+ * @param string $logGroup
3189
+ */
3190
+ public function warning($message, $logGroup = null) {
3191
+ $this->log($message, La_Log::WARNING, $logGroup);
3192
+ }
3193
+
3194
+ /**
3195
+ * logs error message
3196
+ *
3197
+ * @param string $message
3198
+ * @param string $logGroup
3199
+ */
3200
+ public function error($message, $logGroup = null) {
3201
+ $this->log($message, La_Log::ERROR, $logGroup);
3202
+ }
3203
+
3204
+ /**
3205
+ * logs critical error message
3206
+ *
3207
+ * @param string $message
3208
+ * @param string $logGroup
3209
+ */
3210
+ public function critical($message, $logGroup = null) {
3211
+ $this->log($message, La_Log::CRITICAL, $logGroup);
3212
+ }
3213
+
3214
+ public function disableType($type) {
3215
+ $this->disabledTypes[$type] =$type;
3216
+ }
3217
+
3218
+ public function enableAllTypes() {
3219
+ $this->disabledTypes = array();
3220
+ }
3221
+
3222
+ /**
3223
+ *
3224
+ * @return La_Log_LoggerBase
3225
+ */
3226
+ private function create($type) {
3227
+ switch($type) {
3228
+ case La_Log_LoggerDisplay::TYPE:
3229
+ return new La_Log_LoggerDisplay();
3230
+ case La_Log_LoggerFile::TYPE:
3231
+ return new La_Log_LoggerFile();
3232
+ case La_Log_LoggerDatabase::TYPE:
3233
+ case 'db':
3234
+ return new La_Log_LoggerDatabase();
3235
+ }
3236
+ throw new La_Log_Exception("Log system '$type' does not exist");
3237
+ }
3238
+
3239
+ private function findLogFile() {
3240
+ $calls = debug_backtrace();
3241
+
3242
+ $foundObject = null;
3243
+
3244
+ // special handling for sql benchmarks
3245
+ if($this->sqlBenchmarkFound($calls)) {
3246
+ $foundObject = $this->findFileBySqlBenchmark();
3247
+ }
3248
+
3249
+ if($foundObject == null) {
3250
+ $foundObject = $this->findFileByCallingMethod($calls);
3251
+ }
3252
+ if($foundObject == null) {
3253
+ $foundObject = $this->findLatestObjectBeforeString("Logger.class.php");
3254
+ }
3255
+ if($foundObject == null) {
3256
+ $last = count($calls);
3257
+ $last -= 1;
3258
+ if($last <0) {
3259
+ $last = 0;
3260
+ }
3261
+
3262
+ $foundObject = $calls[$last];
3263
+ }
3264
+
3265
+ return $foundObject;
3266
+ }
3267
+
3268
+ private function sqlBenchmarkFound($calls) {
3269
+ foreach($calls as $obj) {
3270
+ if(isset($obj['function']) && $obj['function'] == "sqlBenchmarkEnd") {
3271
+ return true;
3272
+ }
3273
+ }
3274
+ return false;
3275
+ }
3276
+
3277
+ private function findFileBySqlBenchmark() {
3278
+ $foundFile = $this->findLatestObjectBeforeString("DbEngine");
3279
+ if($foundFile != null && is_object($foundFile['object'])) {
3280
+ $foundFile['classVariables'] = $this->getObjectVariables($foundFile['object']);
3281
+ }
3282
+ return $foundFile;
3283
+ }
3284
+
3285
+ private function getObjectVariables($object) {
3286
+ if(is_object($object)) {
3287
+ $class = get_class($object);
3288
+ $methods = get_class_methods($class);
3289
+ if(in_array("__toString", $methods)) {
3290
+ return $object->__toString();
3291
+ }
3292
+ }
3293
+ return '';
3294
+ }
3295
+
3296
+ private function findFileByCallingMethod($calls) {
3297
+ $functionNames = array('debug', 'info', 'warning', 'error', 'critical', 'log');
3298
+ $foundObject = null;
3299
+ foreach($functionNames as $name) {
3300
+ $foundObject = $this->findCallingFile($calls, $name);
3301
+ if($foundObject != null) {
3302
+ return $foundObject;
3303
+ }
3304
+ }
3305
+
3306
+ return null;
3307
+ }
3308
+
3309
+ private function findCallingFile($calls, $functionName) {
3310
+ foreach($calls as $obj) {
3311
+ if(isset($obj['function']) && $obj['function'] == $functionName) {
3312
+ return $obj;
3313
+ }
3314
+ }
3315
+
3316
+ return null;
3317
+ }
3318
+
3319
+ private function findLatestObjectBeforeString($text) {
3320
+ $callsReversed = array_reverse( debug_backtrace() );
3321
+
3322
+ $lastObject = null;
3323
+ foreach($callsReversed as $obj) {
3324
+ if(!isset($obj['file'])) {
3325
+ continue;
3326
+ }
3327
+ $pos = strpos($obj['file'], $text);
3328
+ if($pos !== false && $lastObject != null) {
3329
+ return $lastObject;
3330
+ }
3331
+ $lastObject = $obj;
3332
+ }
3333
+ return null;
3334
+ }
3335
+ }
3336
+
3337
+ } //end La_Log_Logger
3338
+
3339
+ if (!class_exists('La_Api_IncompatibleVersionException', false)) {
3340
+ class La_Api_IncompatibleVersionException extends Exception {
3341
+
3342
+ private $apiLink;
3343
+
3344
+ public function __construct($url) {
3345
+ $this->apiLink = $url. '?C=La_Api_DownloadAPI&M=download&FormRequest=Y&FormResponse=Y';
3346
+ parent::__construct('Version of API not corresponds to the Application version. Please <a href="' . $this->apiLink . '">download latest version of API</a>.', 0);
3347
+ }
3348
+
3349
+ public function getApiDownloadLink() {
3350
+ return $this->apiLink;
3351
+ }
3352
+
3353
+ }
3354
+
3355
+ } //end La_Api_IncompatibleVersionException
3356
+
3357
+ if (!class_exists('La_Api_Session', false)) {
3358
+ class La_Api_Session extends La_Object {
3359
+ const MERCHANT = 'M';
3360
+ const AFFILIATE = 'A';
3361
+
3362
+ private $url;
3363
+ private $sessionId = '';
3364
+ private $debug = false;
3365
+ private $message = '';
3366
+ private $roleType = '';
3367
+
3368
+ public function __construct($url) {
3369
+ $this->url = $url;
3370
+ }
3371
+ /**
3372
+ *
3373
+ * @param $username
3374
+ * @param $password
3375
+ * @param $roleType La_Api_Session::MERCHANT or La_Api_Session::AFFILIATE
3376
+ * @param $languageCode language code (e.g. en-US, de-DE, sk, cz, du, ...)
3377
+ * @return boolean true if user was sucesfully logged
3378
+ * @throws La_Api_IncompatibleVersionException
3379
+ */
3380
+ public function login($username, $password, $roleType = self::MERCHANT, $languageCode = null) {
3381
+ $request = new La_Rpc_FormRequest("La_Api_AuthService", "authenticate");
3382
+ $request->setUrl($this->url);
3383
+ $request->setField("username", $username);
3384
+ $request->setField("password", $password);
3385
+ $request->setField("roleType", $roleType);
3386
+ $request->setField('apiVersion', self::getAPIVersion());
3387
+ if($languageCode != null) {
3388
+ $request->setField("language", $languageCode);
3389
+ }
3390
+
3391
+ $this->roleType = $roleType;
3392
+
3393
+ try {
3394
+ $request->sendNow();
3395
+ } catch(Exception $e) {
3396
+ $this->setMessage("Connection error: ".$e->getMessage());
3397
+ return false;
3398
+ }
3399
+
3400
+ $form = $request->getForm();
3401
+ $this->checkApiVersion($form);
3402
+
3403
+ $this->message = $form->getInfoMessage();
3404
+
3405
+ if($form->isSuccessful() && $form->existsField("S")) {
3406
+ $this->sessionId = $form->getFieldValue("S");
3407
+ $this->setMessage($form->getInfoMessage());
3408
+ return true;
3409
+ }
3410
+
3411
+ $this->setMessage($form->getErrorMessage());
3412
+ return false;
3413
+ }
3414
+
3415
+ /**
3416
+ * Get version of installed application
3417
+ *
3418
+ * @return string version of installed application
3419
+ */
3420
+ public function getAppVersion() {
3421
+ $request = new La_Rpc_FormRequest("La_Api_AuthService", "getAppVersion");
3422
+ $request->setUrl($this->url);
3423
+
3424
+ try {
3425
+ $request->sendNow();
3426
+ } catch(Exception $e) {
3427
+ $this->setMessage("Connection error: ".$e->getMessage());
3428
+ return false;
3429
+ }
3430
+
3431
+ $form = $request->getForm();
3432
+ return $form->getFieldValue('version');
3433
+ }
3434
+
3435
+
3436
+ public function getMessage() {
3437
+ return $this->message;
3438
+ }
3439
+
3440
+ private function setMessage($msg) {
3441
+ $this->message = $msg;
3442
+ }
3443
+
3444
+ public function getDebug() {
3445
+ return $this->debug;
3446
+ }
3447
+
3448
+ public function setDebug($debug = true) {
3449
+ $this->debug = $debug;
3450
+ }
3451
+
3452
+ public function getSessionId() {
3453
+ return $this->sessionId;
3454
+ }
3455
+
3456
+ public function setSessionId($id) {
3457
+ $this->sessionId = $id;
3458
+ }
3459
+
3460
+ public function getRoleType() {
3461
+ return $this->roleType;
3462
+ }
3463
+
3464
+ public function getUrl() {
3465
+ return $this->url;
3466
+ }
3467
+
3468
+ public function getUrlWithSessionInfo($url) {
3469
+ if (strpos($url, '?') === false) {
3470
+ return $url . '?S=' . $this->getSessionId();
3471
+ }
3472
+ return $url . '&S=' . $this->getSessionId();
3473
+ }
3474
+
3475
+ /**
3476
+ * @param $latestVersion
3477
+ * @throws La_Api_IncompatibleVersionException
3478
+ */
3479
+ private function checkApiVersion(La_Rpc_Form $form) {
3480
+ if ($form->getFieldValue('correspondsApi') === Gpf::NO) {
3481
+ throw new La_Api_IncompatibleVersionException($this->url);
3482
+ }
3483
+ }
3484
+
3485
+ /**
3486
+ * @return String
3487
+ */
3488
+ public static function getAPIVersion($fileName = __FILE__) {
3489
+ $fileHandler = fopen($fileName, 'r');
3490
+ fseek($fileHandler, -6 -32, SEEK_END);
3491
+ $hash = fgets($fileHandler);
3492
+ return substr($hash, 0, -1);
3493
+ }
3494
+ }
3495
+
3496
+ } //end La_Api_Session
3497
+
3498
+ if (!class_exists('La_Rpc_Json', false)) {
3499
+ class La_Rpc_Json implements La_Rpc_DataEncoder, La_Rpc_DataDecoder {
3500
+ /**
3501
+ * Marker constant for Services_JSON::decode(), used to flag stack state
3502
+ */
3503
+ const SERVICES_JSON_SLICE = 1;
3504
+
3505
+ /**
3506
+ * Marker constant for Services_JSON::decode(), used to flag stack state
3507
+ */
3508
+ const SERVICES_JSON_IN_STR = 2;
3509
+
3510
+ /**
3511
+ * Marker constant for Services_JSON::decode(), used to flag stack state
3512
+ */
3513
+ const SERVICES_JSON_IN_ARR = 3;
3514
+
3515
+ /**
3516
+ * Marker constant for Services_JSON::decode(), used to flag stack state
3517
+ */
3518
+ const SERVICES_JSON_IN_OBJ = 4;
3519
+
3520
+ /**
3521
+ * Marker constant for Services_JSON::decode(), used to flag stack state
3522
+ */
3523
+ const SERVICES_JSON_IN_CMT = 5;
3524
+
3525
+ /**
3526
+ * Behavior switch for Services_JSON::decode()
3527
+ */
3528
+ const SERVICES_JSON_LOOSE_TYPE = 16;
3529
+
3530
+ /**
3531
+ * Behavior switch for Services_JSON::decode()
3532
+ */
3533
+ const SERVICES_JSON_SUPPRESS_ERRORS = 32;
3534
+
3535
+ /**
3536
+ * constructs a new JSON instance
3537
+ *
3538
+ * @param int $use object behavior flags; combine with boolean-OR
3539
+ *
3540
+ * possible values:
3541
+ * - SERVICES_JSON_LOOSE_TYPE: loose typing.
3542
+ * "{...}" syntax creates associative arrays
3543
+ * instead of objects in decode().
3544
+ * - SERVICES_JSON_SUPPRESS_ERRORS: error suppression.
3545
+ * Values which can't be encoded (e.g. resources)
3546
+ * appear as NULL instead of throwing errors.
3547
+ * By default, a deeply-nested resource will
3548
+ * bubble up with an error, so all return values
3549
+ * from encode() should be checked with isError()
3550
+ */
3551
+ function __construct($use = 0)
3552
+ {
3553
+ $this->use = $use;
3554
+ }
3555
+
3556
+ /**
3557
+ * @var La_Rpc_Json
3558
+ */
3559
+ private static $instance;
3560
+
3561
+ /**
3562
+ *
3563
+ * @return La_Rpc_Json
3564
+ */
3565
+ private function getInstance() {
3566
+ if (self::$instance === null) {
3567
+ self::$instance = new self;
3568
+ }
3569
+ return self::$instance;
3570
+ }
3571
+
3572
+ public static function encodeStatic($var) {
3573
+ return self::getInstance()->encode($var);
3574
+ }
3575
+
3576
+ public static function decodeStatic($var) {
3577
+ return self::getInstance()->decode($var);
3578
+ }
3579
+
3580
+ /**
3581
+ * convert a string from one UTF-16 char to one UTF-8 char
3582
+ *
3583
+ * Normally should be handled by mb_convert_encoding, but
3584
+ * provides a slower PHP-only method for installations
3585
+ * that lack the multibye string extension.
3586
+ *
3587
+ * @param string $utf16 UTF-16 character
3588
+ * @return string UTF-8 character
3589
+ * @access private
3590
+ */
3591
+ function utf162utf8($utf16)
3592
+ {
3593
+ // oh please oh please oh please oh please oh please
3594
+ if(La_Php::isFunctionEnabled('mb_convert_encoding')) {
3595
+ return mb_convert_encoding($utf16, 'UTF-8', 'UTF-16');
3596
+ }
3597
+
3598
+ $bytes = (ord($utf16{0}) << 8) | ord($utf16{1});
3599
+
3600
+ switch(true) {
3601
+ case ((0x7F & $bytes) == $bytes):
3602
+ // this case should never be reached, because we are in ASCII range
3603
+ // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3604
+ return chr(0x7F & $bytes);
3605
+
3606
+ case (0x07FF & $bytes) == $bytes:
3607
+ // return a 2-byte UTF-8 character
3608
+ // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3609
+ return chr(0xC0 | (($bytes >> 6) & 0x1F))
3610
+ . chr(0x80 | ($bytes & 0x3F));
3611
+
3612
+ case (0xFFFF & $bytes) == $bytes:
3613
+ // return a 3-byte UTF-8 character
3614
+ // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3615
+ return chr(0xE0 | (($bytes >> 12) & 0x0F))
3616
+ . chr(0x80 | (($bytes >> 6) & 0x3F))
3617
+ . chr(0x80 | ($bytes & 0x3F));
3618
+ }
3619
+
3620
+ // ignoring UTF-32 for now, sorry
3621
+ return '';
3622
+ }
3623
+
3624
+ /**
3625
+ * convert a string from one UTF-8 char to one UTF-16 char
3626
+ *
3627
+ * Normally should be handled by mb_convert_encoding, but
3628
+ * provides a slower PHP-only method for installations
3629
+ * that lack the multibye string extension.
3630
+ *
3631
+ * @param string $utf8 UTF-8 character
3632
+ * @return string UTF-16 character
3633
+ * @access private
3634
+ */
3635
+ function utf82utf16($utf8)
3636
+ {
3637
+ // oh please oh please oh please oh please oh please
3638
+ if(La_Php::isFunctionEnabled('mb_convert_encoding')) {
3639
+ return mb_convert_encoding($utf8, 'UTF-16', 'UTF-8');
3640
+ }
3641
+
3642
+ switch(strlen($utf8)) {
3643
+ case 1:
3644
+ // this case should never be reached, because we are in ASCII range
3645
+ // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3646
+ return $utf8;
3647
+
3648
+ case 2:
3649
+ // return a UTF-16 character from a 2-byte UTF-8 char
3650
+ // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3651
+ return chr(0x07 & (ord($utf8{0}) >> 2))
3652
+ . chr((0xC0 & (ord($utf8{0}) << 6))
3653
+ | (0x3F & ord($utf8{1})));
3654
+
3655
+ case 3:
3656
+ // return a UTF-16 character from a 3-byte UTF-8 char
3657
+ // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3658
+ return chr((0xF0 & (ord($utf8{0}) << 4))
3659
+ | (0x0F & (ord($utf8{1}) >> 2)))
3660
+ . chr((0xC0 & (ord($utf8{1}) << 6))
3661
+ | (0x7F & ord($utf8{2})));
3662
+ }
3663
+
3664
+ // ignoring UTF-32 for now, sorry
3665
+ return '';
3666
+ }
3667
+
3668
+ public function encodeResponse(La_Rpc_Serializable $response) {
3669
+ return $this->encode($response->toObject());
3670
+ }
3671
+
3672
+ /**
3673
+ * encodes an arbitrary variable into JSON format
3674
+ *
3675
+ * @param mixed $var any number, boolean, string, array, or object to be encoded.
3676
+ * see argument 1 to Services_JSON() above for array-parsing behavior.
3677
+ * if var is a strng, note that encode() always expects it
3678
+ * to be in ASCII or UTF-8 format!
3679
+ *
3680
+ * @return mixed JSON string representation of input var or an error if a problem occurs
3681
+ * @access public
3682
+ */
3683
+ public function encode($var) {
3684
+ if ($this->isJsonEncodeEnabled()) {
3685
+ return @json_encode($var);
3686
+ }
3687
+ switch (gettype($var)) {
3688
+ case 'boolean':
3689
+ return $var ? 'true' : 'false';
3690
+
3691
+ case 'NULL':
3692
+ return 'null';
3693
+
3694
+ case 'integer':
3695
+ return (int) $var;
3696
+
3697
+ case 'double':
3698
+ case 'float':
3699
+ return (float) $var;
3700
+
3701
+ case 'string':
3702
+ // STRINGS ARE EXPECTED TO BE IN ASCII OR UTF-8 FORMAT
3703
+ $ascii = '';
3704
+ $strlen_var = strlen($var);
3705
+
3706
+ /*
3707
+ * Iterate over every character in the string,
3708
+ * escaping with a slash or encoding to UTF-8 where necessary
3709
+ */
3710
+ for ($c = 0; $c < $strlen_var; ++$c) {
3711
+
3712
+ $ord_var_c = ord($var{$c});
3713
+
3714
+ switch (true) {
3715
+ case $ord_var_c == 0x08:
3716
+ $ascii .= '\b';
3717
+ break;
3718
+ case $ord_var_c == 0x09:
3719
+ $ascii .= '\t';
3720
+ break;
3721
+ case $ord_var_c == 0x0A:
3722
+ $ascii .= '\n';
3723
+ break;
3724
+ case $ord_var_c == 0x0C:
3725
+ $ascii .= '\f';
3726
+ break;
3727
+ case $ord_var_c == 0x0D:
3728
+ $ascii .= '\r';
3729
+ break;
3730
+
3731
+ case $ord_var_c == 0x22:
3732
+ case $ord_var_c == 0x2F:
3733
+ case $ord_var_c == 0x5C:
3734
+ // double quote, slash, slosh
3735
+ $ascii .= '\\'.$var{$c};
3736
+ break;
3737
+
3738
+ case (($ord_var_c >= 0x20) && ($ord_var_c <= 0x7F)):
3739
+ // characters U-00000000 - U-0000007F (same as ASCII)
3740
+ $ascii .= $var{$c};
3741
+ break;
3742
+
3743
+ case (($ord_var_c & 0xE0) == 0xC0):
3744
+ // characters U-00000080 - U-000007FF, mask 1 1 0 X X X X X
3745
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3746
+ $char = pack('C*', $ord_var_c, ord($var{$c + 1}));
3747
+ $c += 1;
3748
+ $utf16 = $this->utf82utf16($char);
3749
+ $ascii .= sprintf('\u%04s', bin2hex($utf16));
3750
+ break;
3751
+
3752
+ case (($ord_var_c & 0xF0) == 0xE0):
3753
+ // characters U-00000800 - U-0000FFFF, mask 1 1 1 0 X X X X
3754
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3755
+ $char = pack('C*', $ord_var_c,
3756
+ ord($var{$c + 1}),
3757
+ ord($var{$c + 2}));
3758
+ $c += 2;
3759
+ $utf16 = $this->utf82utf16($char);
3760
+ $ascii .= sprintf('\u%04s', bin2hex($utf16));
3761
+ break;
3762
+
3763
+ case (($ord_var_c & 0xF8) == 0xF0):
3764
+ // characters U-00010000 - U-001FFFFF, mask 1 1 1 1 0 X X X
3765
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3766
+ $char = pack('C*', $ord_var_c,
3767
+ ord($var{$c + 1}),
3768
+ ord($var{$c + 2}),
3769
+ ord($var{$c + 3}));
3770
+ $c += 3;
3771
+ $utf16 = $this->utf82utf16($char);
3772
+ $ascii .= sprintf('\u%04s', bin2hex($utf16));
3773
+ break;
3774
+
3775
+ case (($ord_var_c & 0xFC) == 0xF8):
3776
+ // characters U-00200000 - U-03FFFFFF, mask 111110XX
3777
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3778
+ $char = pack('C*', $ord_var_c,
3779
+ ord($var{$c + 1}),
3780
+ ord($var{$c + 2}),
3781
+ ord($var{$c + 3}),
3782
+ ord($var{$c + 4}));
3783
+ $c += 4;
3784
+ $utf16 = $this->utf82utf16($char);
3785
+ $ascii .= sprintf('\u%04s', bin2hex($utf16));
3786
+ break;
3787
+
3788
+ case (($ord_var_c & 0xFE) == 0xFC):
3789
+ // characters U-04000000 - U-7FFFFFFF, mask 1111110X
3790
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
3791
+ $char = pack('C*', $ord_var_c,
3792
+ ord($var{$c + 1}),
3793
+ ord($var{$c + 2}),
3794
+ ord($var{$c + 3}),
3795
+ ord($var{$c + 4}),
3796
+ ord($var{$c + 5}));
3797
+ $c += 5;
3798
+ $utf16 = $this->utf82utf16($char);
3799
+ $ascii .= sprintf('\u%04s', bin2hex($utf16));
3800
+ break;
3801
+ }
3802
+ }
3803
+
3804
+ return '"'.$ascii.'"';
3805
+
3806
+ case 'array':
3807
+ /*
3808
+ * As per JSON spec if any array key is not an integer
3809
+ * we must treat the the whole array as an object. We
3810
+ * also try to catch a sparsely populated associative
3811
+ * array with numeric keys here because some JS engines
3812
+ * will create an array with empty indexes up to
3813
+ * max_index which can cause memory issues and because
3814
+ * the keys, which may be relevant, will be remapped
3815
+ * otherwise.
3816
+ *
3817
+ * As per the ECMA and JSON specification an object may
3818
+ * have any string as a property. Unfortunately due to
3819
+ * a hole in the ECMA specification if the key is a
3820
+ * ECMA reserved word or starts with a digit the
3821
+ * parameter is only accessible using ECMAScript's
3822
+ * bracket notation.
3823
+ */
3824
+
3825
+ // treat as a JSON object
3826
+ if (is_array($var) && count($var) && (array_keys($var) !== range(0, sizeof($var) - 1))) {
3827
+ $properties = array_map(array($this, 'name_value'), array_keys($var), array_values($var));
3828
+
3829
+ foreach($properties as $property) {
3830
+ if(La_Rpc_Json::isError($property)) {
3831
+ return $property;
3832
+ }
3833
+ }
3834
+
3835
+ return '{' . join(',', $properties) . '}';
3836
+ }
3837
+
3838
+ // treat it like a regular array
3839
+ $elements = array_map(array($this, 'encode'), $var);
3840
+
3841
+ foreach($elements as $element) {
3842
+ if(La_Rpc_Json::isError($element)) {
3843
+ return $element;
3844
+ }
3845
+ }
3846
+
3847
+ return '[' . join(',', $elements) . ']';
3848
+
3849
+ case 'object':
3850
+ $vars = get_object_vars($var);
3851
+
3852
+ $properties = array_map(array($this, 'name_value'),
3853
+ array_keys($vars),
3854
+ array_values($vars));
3855
+
3856
+ foreach($properties as $property) {
3857
+ if(La_Rpc_Json::isError($property)) {
3858
+ return $property;
3859
+ }
3860
+ }
3861
+
3862
+ return '{' . join(',', $properties) . '}';
3863
+
3864
+ default:
3865
+ if ($this->use & self::SERVICES_JSON_SUPPRESS_ERRORS) {
3866
+ return 'null';
3867
+ }
3868
+ return new La_Rpc_Json_Error(gettype($var)." can not be encoded as JSON string");
3869
+ }
3870
+ }
3871
+
3872
+ /**
3873
+ * array-walking function for use in generating JSON-formatted name-value pairs
3874
+ *
3875
+ * @param string $name name of key to use
3876
+ * @param mixed $value reference to an array element to be encoded
3877
+ *
3878
+ * @return string JSON-formatted name-value pair, like '"name":value'
3879
+ * @access private
3880
+ */
3881
+ function name_value($name, $value)
3882
+ {
3883
+ $encoded_value = $this->encode($value);
3884
+
3885
+ if(La_Rpc_Json::isError($encoded_value)) {
3886
+ return $encoded_value;
3887
+ }
3888
+
3889
+ return $this->encode(strval($name)) . ':' . $encoded_value;
3890
+ }
3891
+
3892
+ /**
3893
+ * reduce a string by removing leading and trailing comments and whitespace
3894
+ *
3895
+ * @param $str string string value to strip of comments and whitespace
3896
+ *
3897
+ * @return string string value stripped of comments and whitespace
3898
+ * @access private
3899
+ */
3900
+ function reduce_string($str)
3901
+ {
3902
+ $str = preg_replace(array(
3903
+
3904
+ // eliminate single line comments in '// ...' form
3905
+ '#^\s*//(.+)$#m',
3906
+
3907
+ // eliminate multi-line comments in '/* ... */' form, at start of string
3908
+ '#^\s*/\*(.+)\*/#Us',
3909
+
3910
+ // eliminate multi-line comments in '/* ... */' form, at end of string
3911
+ '#/\*(.+)\*/\s*$#Us'
3912
+
3913
+ ), '', $str);
3914
+
3915
+ // eliminate extraneous space
3916
+ return trim($str);
3917
+ }
3918
+
3919
+ /**
3920
+ * decodes a JSON string into appropriate variable
3921
+ *
3922
+ * @param string $str JSON-formatted string
3923
+ *
3924
+ * @return mixed number, boolean, string, array, or object
3925
+ * corresponding to given JSON input string.
3926
+ * See argument 1 to Services_JSON() above for object-output behavior.
3927
+ * Note that decode() always returns strings
3928
+ * in ASCII or UTF-8 format!
3929
+ * @access public
3930
+ */
3931
+ function decode($str)
3932
+ {
3933
+ if ($this->isJsonDecodeEnabled()) {
3934
+ return json_decode($str);
3935
+ }
3936
+
3937
+ $str = $this->reduce_string($str);
3938
+
3939
+ switch (strtolower($str)) {
3940
+ case 'true':
3941
+ return true;
3942
+
3943
+ case 'false':
3944
+ return false;
3945
+
3946
+ case 'null':
3947
+ return null;
3948
+
3949
+ default:
3950
+ $m = array();
3951
+
3952
+ if (is_numeric($str)) {
3953
+ // Lookie-loo, it's a number
3954
+
3955
+ // This would work on its own, but I'm trying to be
3956
+ // good about returning integers where appropriate:
3957
+ // return (float)$str;
3958
+
3959
+ // Return float or int, as appropriate
3960
+ return ((float)$str == (integer)$str)
3961
+ ? (integer)$str
3962
+ : (float)$str;
3963
+
3964
+ } elseif (preg_match('/^("|\').*(\1)$/s', $str, $m) && $m[1] == $m[2]) {
3965
+ // STRINGS RETURNED IN UTF-8 FORMAT
3966
+ $delim = substr($str, 0, 1);
3967
+ $chrs = substr($str, 1, -1);
3968
+ $utf8 = '';
3969
+ $strlen_chrs = strlen($chrs);
3970
+
3971
+ for ($c = 0; $c < $strlen_chrs; ++$c) {
3972
+
3973
+ $substr_chrs_c_2 = substr($chrs, $c, 2);
3974
+ $ord_chrs_c = ord($chrs{$c});
3975
+
3976
+ switch (true) {
3977
+ case $substr_chrs_c_2 == '\b':
3978
+ $utf8 .= chr(0x08);
3979
+ ++$c;
3980
+ break;
3981
+ case $substr_chrs_c_2 == '\t':
3982
+ $utf8 .= chr(0x09);
3983
+ ++$c;
3984
+ break;
3985
+ case $substr_chrs_c_2 == '\n':
3986
+ $utf8 .= chr(0x0A);
3987
+ ++$c;
3988
+ break;
3989
+ case $substr_chrs_c_2 == '\f':
3990
+ $utf8 .= chr(0x0C);
3991
+ ++$c;
3992
+ break;
3993
+ case $substr_chrs_c_2 == '\r':
3994
+ $utf8 .= chr(0x0D);
3995
+ ++$c;
3996
+ break;
3997
+
3998
+ case $substr_chrs_c_2 == '\\"':
3999
+ case $substr_chrs_c_2 == '\\\'':
4000
+ case $substr_chrs_c_2 == '\\\\':
4001
+ case $substr_chrs_c_2 == '\\/':
4002
+ if (($delim == '"' && $substr_chrs_c_2 != '\\\'') ||
4003
+ ($delim == "'" && $substr_chrs_c_2 != '\\"')) {
4004
+ $utf8 .= $chrs{++$c};
4005
+ }
4006
+ break;
4007
+
4008
+ case preg_match('/\\\u[0-9A-F]{4}/i', substr($chrs, $c, 6)):
4009
+ // single, escaped unicode character
4010
+ $utf16 = chr(hexdec(substr($chrs, ($c + 2), 2)))
4011
+ . chr(hexdec(substr($chrs, ($c + 4), 2)));
4012
+ $utf8 .= $this->utf162utf8($utf16);
4013
+ $c += 5;
4014
+ break;
4015
+
4016
+ case ($ord_chrs_c >= 0x20) && ($ord_chrs_c <= 0x7F):
4017
+ $utf8 .= $chrs{$c};
4018
+ break;
4019
+
4020
+ case ($ord_chrs_c & 0xE0) == 0xC0:
4021
+ // characters U-00000080 - U-000007FF, mask 1 1 0 X X X X X
4022
+ //see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
4023
+ $utf8 .= substr($chrs, $c, 2);
4024
+ ++$c;
4025
+ break;
4026
+
4027
+ case ($ord_chrs_c & 0xF0) == 0xE0:
4028
+ // characters U-00000800 - U-0000FFFF, mask 1 1 1 0 X X X X
4029
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
4030
+ $utf8 .= substr($chrs, $c, 3);
4031
+ $c += 2;
4032
+ break;
4033
+
4034
+ case ($ord_chrs_c & 0xF8) == 0xF0:
4035
+ // characters U-00010000 - U-001FFFFF, mask 1 1 1 1 0 X X X
4036
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
4037
+ $utf8 .= substr($chrs, $c, 4);
4038
+ $c += 3;
4039
+ break;
4040
+
4041
+ case ($ord_chrs_c & 0xFC) == 0xF8:
4042
+ // characters U-00200000 - U-03FFFFFF, mask 111110XX
4043
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
4044
+ $utf8 .= substr($chrs, $c, 5);
4045
+ $c += 4;
4046
+ break;
4047
+
4048
+ case ($ord_chrs_c & 0xFE) == 0xFC:
4049
+ // characters U-04000000 - U-7FFFFFFF, mask 1111110X
4050
+ // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
4051
+ $utf8 .= substr($chrs, $c, 6);
4052
+ $c += 5;
4053
+ break;
4054
+
4055
+ }
4056
+
4057
+ }
4058
+
4059
+ return $utf8;
4060
+
4061
+ } elseif (preg_match('/^\[.*\]$/s', $str) || preg_match('/^\{.*\}$/s', $str)) {
4062
+ // array, or object notation
4063
+
4064
+ if ($str{0} == '[') {
4065
+ $stk = array(self::SERVICES_JSON_IN_ARR);
4066
+ $arr = array();
4067
+ } else {
4068
+ if ($this->use & self::SERVICES_JSON_LOOSE_TYPE) {
4069
+ $stk = array(self::SERVICES_JSON_IN_OBJ);
4070
+ $obj = array();
4071
+ } else {
4072
+ $stk = array(self::SERVICES_JSON_IN_OBJ);
4073
+ $obj = new stdClass();
4074
+ }
4075
+ }
4076
+
4077
+ array_push($stk, array('what' => self::SERVICES_JSON_SLICE,
4078
+ 'where' => 0,
4079
+ 'delim' => false));
4080
+
4081
+ $chrs = substr($str, 1, -1);
4082
+ $chrs = $this->reduce_string($chrs);
4083
+
4084
+ if ($chrs == '') {
4085
+ if (reset($stk) == self::SERVICES_JSON_IN_ARR) {
4086
+ return $arr;
4087
+
4088
+ } else {
4089
+ return $obj;
4090
+
4091
+ }
4092
+ }
4093
+
4094
+ //print("\nparsing {$chrs}\n");
4095
+
4096
+ $strlen_chrs = strlen($chrs);
4097
+
4098
+ for ($c = 0; $c <= $strlen_chrs; ++$c) {
4099
+
4100
+ $top = end($stk);
4101
+ $substr_chrs_c_2 = substr($chrs, $c, 2);
4102
+
4103
+ if (($c == $strlen_chrs) || (($chrs{$c} == ',') && ($top['what'] == self::SERVICES_JSON_SLICE))) {
4104
+ // found a comma that is not inside a string, array, etc.,
4105
+ // OR we've reached the end of the character list
4106
+ $slice = substr($chrs, $top['where'], ($c - $top['where']));
4107
+ array_push($stk, array('what' => self::SERVICES_JSON_SLICE, 'where' => ($c + 1), 'delim' => false));
4108
+ //print("Found split at {$c}: ".substr($chrs, $top['where'], (1 + $c - $top['where']))."\n");
4109
+
4110
+ if (reset($stk) == self::SERVICES_JSON_IN_ARR) {
4111
+ // we are in an array, so just push an element onto the stack
4112
+ array_push($arr, $this->decode($slice));
4113
+
4114
+ } elseif (reset($stk) == self::SERVICES_JSON_IN_OBJ) {
4115
+ // we are in an object, so figure
4116
+ // out the property name and set an
4117
+ // element in an associative array,
4118
+ // for now
4119
+ $parts = array();
4120
+
4121
+ if (preg_match('/^\s*(["\'].*[^\\\]["\'])\s*:\s*(\S.*),?$/Uis', $slice, $parts)) {
4122
+ // "name":value pair
4123
+ $key = $this->decode($parts[1]);
4124
+ $val = $this->decode($parts[2]);
4125
+
4126
+ if ($this->use & self::SERVICES_JSON_LOOSE_TYPE) {
4127
+ $obj[$key] = $val;
4128
+ } else {
4129
+ $obj->$key = $val;
4130
+ }
4131
+ } elseif (preg_match('/^\s*(\w+)\s*:\s*(\S.*),?$/Uis', $slice, $parts)) {
4132
+ // name:value pair, where name is unquoted
4133
+ $key = $parts[1];
4134
+ $val = $this->decode($parts[2]);
4135
+
4136
+ if ($this->use & self::SERVICES_JSON_LOOSE_TYPE) {
4137
+ $obj[$key] = $val;
4138
+ } else {
4139
+ $obj->$key = $val;
4140
+ }
4141
+ }
4142
+
4143
+ }
4144
+
4145
+ } elseif ((($chrs{$c} == '"') || ($chrs{$c} == "'")) && ($top['what'] != self::SERVICES_JSON_IN_STR)) {
4146
+ // found a quote, and we are not inside a string
4147
+ array_push($stk, array('what' => self::SERVICES_JSON_IN_STR, 'where' => $c, 'delim' => $chrs{$c}));
4148
+ //print("Found start of string at {$c}\n");
4149
+
4150
+ } elseif (($chrs{$c} == $top['delim']) &&
4151
+ ($top['what'] == self::SERVICES_JSON_IN_STR) &&
4152
+ (($chrs{$c - 1} != '\\') ||
4153
+ ($chrs{$c - 1} == '\\' && $chrs{$c - 2} == '\\'))) {
4154
+ // found a quote, we're in a string, and it's not escaped
4155
+ array_pop($stk);
4156
+ //print("Found end of string at {$c}: ".substr($chrs, $top['where'], (1 + 1 + $c - $top['where']))."\n");
4157
+
4158
+ } elseif (($chrs{$c} == '[') &&
4159
+ in_array($top['what'], array(self::SERVICES_JSON_SLICE, self::SERVICES_JSON_IN_ARR, self::SERVICES_JSON_IN_OBJ))) {
4160
+ // found a left-bracket, and we are in an array, object, or slice
4161
+ array_push($stk, array('what' => self::SERVICES_JSON_IN_ARR, 'where' => $c, 'delim' => false));
4162
+ //print("Found start of array at {$c}\n");
4163
+
4164
+ } elseif (($chrs{$c} == ']') && ($top['what'] == self::SERVICES_JSON_IN_ARR)) {
4165
+ // found a right-bracket, and we're in an array
4166
+ array_pop($stk);
4167
+ //print("Found end of array at {$c}: ".substr($chrs, $top['where'], (1 + $c - $top['where']))."\n");
4168
+
4169
+ } elseif (($chrs{$c} == '{') &&
4170
+ in_array($top['what'], array(self::SERVICES_JSON_SLICE, self::SERVICES_JSON_IN_ARR, self::SERVICES_JSON_IN_OBJ))) {
4171
+ // found a left-brace, and we are in an array, object, or slice
4172
+ array_push($stk, array('what' => self::SERVICES_JSON_IN_OBJ, 'where' => $c, 'delim' => false));
4173
+ //print("Found start of object at {$c}\n");
4174
+
4175
+ } elseif (($chrs{$c} == '}') && ($top['what'] == self::SERVICES_JSON_IN_OBJ)) {
4176
+ // found a right-brace, and we're in an object
4177
+ array_pop($stk);
4178
+ //print("Found end of object at {$c}: ".substr($chrs, $top['where'], (1 + $c - $top['where']))."\n");
4179
+
4180
+ } elseif (($substr_chrs_c_2 == '/*') &&
4181
+ in_array($top['what'], array(self::SERVICES_JSON_SLICE, self::SERVICES_JSON_IN_ARR, self::SERVICES_JSON_IN_OBJ))) {
4182
+ // found a comment start, and we are in an array, object, or slice
4183
+ array_push($stk, array('what' => self::SERVICES_JSON_IN_CMT, 'where' => $c, 'delim' => false));
4184
+ $c++;
4185
+ //print("Found start of comment at {$c}\n");
4186
+
4187
+ } elseif (($substr_chrs_c_2 == '*/') && ($top['what'] == self::SERVICES_JSON_IN_CMT)) {
4188
+ // found a comment end, and we're in one now
4189
+ array_pop($stk);
4190
+ $c++;
4191
+
4192
+ for ($i = $top['where']; $i <= $c; ++$i)
4193
+ $chrs = substr_replace($chrs, ' ', $i, 1);
4194
+
4195
+ //print("Found end of comment at {$c}: ".substr($chrs, $top['where'], (1 + $c - $top['where']))."\n");
4196
+
4197
+ }
4198
+
4199
+ }
4200
+
4201
+ if (reset($stk) == self::SERVICES_JSON_IN_ARR) {
4202
+ return $arr;
4203
+
4204
+ } elseif (reset($stk) == self::SERVICES_JSON_IN_OBJ) {
4205
+ return $obj;
4206
+
4207
+ }
4208
+
4209
+ }
4210
+ }
4211
+ }
4212
+
4213
+ protected function isJsonEncodeEnabled() {
4214
+ return La_Php::isFunctionEnabled('json_encode');
4215
+ }
4216
+
4217
+ protected function isJsonDecodeEnabled() {
4218
+ return La_Php::isFunctionEnabled('json_decode');
4219
+ }
4220
+
4221
+
4222
+ /**
4223
+ * @todo Ultimately, this should just call PEAR::isError()
4224
+ */
4225
+ function isError($data, $code = null)
4226
+ {
4227
+ if (is_object($data) &&
4228
+ (get_class($data) == 'La_Rpc_Json_Error' || is_subclass_of($data, 'La_Rpc_Json_Error'))) {
4229
+ return true;
4230
+ }
4231
+ return false;
4232
+ }
4233
+ }
4234
+
4235
+ class La_Rpc_Json_Error {
4236
+ private $message;
4237
+
4238
+ public function __construct($message) {
4239
+ $this->message = $message;
4240
+ }
4241
+ }
4242
+
4243
+
4244
+ } //end La_Rpc_Json
4245
+
4246
+ if (!class_exists('La_Rpc_JsonObject', false)) {
4247
+ class La_Rpc_JsonObject extends La_Object {
4248
+
4249
+ public function __construct($object = null) {
4250
+ if ($object != null) {
4251
+ $this->initFrom($object);
4252
+ }
4253
+ }
4254
+
4255
+ public function decode($string) {
4256
+ if ($string == null || $string == "") {
4257
+ throw new La_Exception("Invalid format (".get_class($this).")");
4258
+ }
4259
+ $string = stripslashes($string);
4260
+ $json = new La_Rpc_Json();
4261
+ $object = $json->decode($string);
4262
+ if (!is_object($object)) {
4263
+ throw new La_Exception("Invalid format (".get_class($this).")");
4264
+ }
4265
+ $this->initFrom($object);
4266
+ }
4267
+
4268
+ private function initFrom($object) {
4269
+ $object_vars = get_object_vars($object);
4270
+ foreach ($object_vars as $name => $value) {
4271
+ if (property_exists($this, $name)) {
4272
+ $this->$name = $value;
4273
+ }
4274
+ }
4275
+ }
4276
+
4277
+ public function encode() {
4278
+ $json = new La_Rpc_Json();
4279
+ return $json->encode($this);
4280
+ }
4281
+
4282
+ public function __toString() {
4283
+ return $this->encode();
4284
+ }
4285
+ }
4286
+
4287
+ } //end La_Rpc_JsonObject
4288
+
4289
+ if (!class_exists('La_Net_Http_Client', false)) {
4290
+ class La_Net_Http_Client extends La_Net_Http_ClientBase {
4291
+
4292
+ protected function isNetworkingEnabled() {
4293
+ return Gpf::YES;
4294
+ }
4295
+
4296
+ }
4297
+
4298
+ } //end La_Net_Http_Client
4299
+
4300
+ if (!class_exists('Gpf', false)) {
4301
+ class Gpf {
4302
+ const YES = 'Y';
4303
+ const NO = 'N';
4304
+ }
4305
+ }
4306
+ /*
4307
+ VERSION
4308
+ 8420bb3c08118181a7f384fda817c252
4309
+ */
4310
+ ?>
app/code/local/Qualityunit/Liveagent/Helper/Settings.php ADDED
@@ -0,0 +1,187 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @copyright Copyright (c) 2007 Quality Unit s.r.o.
4
+ * @author Juraj Simon
5
+ * @package WpLiveAgentPlugin
6
+ * @version 1.0.0
7
+ *
8
+ * Licensed under GPL2
9
+ */
10
+
11
+ class Qualityunit_Liveagent_Helper_Settings {
12
+ const CACHE_VALIDITY = 600;
13
+
14
+ //internal settings
15
+ const INTERNAL_SETTINGS = 'la-settings_internal-settings';
16
+ const OWNER_SESSIONID = 'la-settings_owner-sessionid';
17
+ const OWNER_AUTHTOKEN = 'la-settings_owner-authtoken';
18
+ const BUTTONS_DATA = 'la-settings_buttonsdata';
19
+ const ACCOUNT_STATUS = 'la-settings_accountstatus';
20
+ const ACCOUNT_NOT_REACHABLE_TIMES = 'la-settings_accountnotreachabletimes';
21
+ const LAST_BUTTONS_RELOAD_TIME = 'la-settings_lastbuttonsreloadtime';
22
+
23
+
24
+ //general page
25
+ const GENERAL_SETTINGS_PAGE_NAME = 'la-config-general-page';
26
+ const SIGNUP_SETTINGS_PAGE_NAME = 'la-config-signup-page';
27
+ const SIGNUP_WAIT_SETTINGS_PAGE_NAME = 'la-config-signup-wait-page';
28
+
29
+ const LA_URL_SETTING_NAME = 'la-url';
30
+ const LA_OWNER_EMAIL_SETTING_NAME = 'la-owner-email';
31
+ const LA_OWNER_PASSWORD_SETTING_NAME = 'la-owner-password';
32
+ const GENERAL_SETTINGS_PAGE_STATE_SETTING_NAME = 'general-settings-state';
33
+
34
+ //buttons options
35
+ const BUTTONS_SETTINGS_PAGE_NAME = 'la-config-buttons-page';
36
+ const BUTTONS_CONFIGURATION_SETTING_NAME = 'la-buttons-configuration';
37
+
38
+ const NO_AUTH_TOKEN = 'no_auth_token';
39
+
40
+ const ACCOUNT_STATUS_NOTSET = 'N';
41
+ const ACCOUNT_STATUS_SET = 'S';
42
+
43
+ public function sanitizeUrl($url) {
44
+ if (stripos($url, 'http://')!==false || stripos($url, 'https://')!==false) {
45
+ return $url;
46
+ }
47
+ return 'http://' . $url;
48
+ }
49
+
50
+ public function clearCache() {
51
+ /*update_option(self::OWNER_SESSIONID, '');
52
+ update_option(self::OWNER_AUTHTOKEN, '');
53
+ update_option(self::BUTTONS_DATA, '');*/
54
+ }
55
+
56
+ private function setCachedSetting($code, $value) {
57
+ /*$settings = get_option($code);
58
+ $settingValue = $value . "||" . time();
59
+ if ($settings != '') {
60
+ update_option($code, $settingValue);
61
+ } else {
62
+ add_option($code, $settingValue);
63
+ }*/
64
+ }
65
+
66
+ private function getCachedSetting($code) {
67
+ /*$settings = get_option($code);
68
+ if ($settings == null || trim($settings) == '') {
69
+ throw new liveagent_Exception_SettingNotValid(__(sprintf('Setting %s not defined yet.', $code)));
70
+ }
71
+ $settings = explode("||", $settings, 2);
72
+ $validTo = $settings[1] + self::CACHE_VALIDITY + 0;
73
+ if ($validTo > time()) {
74
+ return $settings[0];
75
+ } else {
76
+ if (array_key_exists('time', $settings)) {
77
+ $message = __(sprintf('Setting\'s %s validity exceeded: %s', $code, $settings['time']));
78
+ } else {
79
+ $message = __(sprintf('Setting\'s %s validity exceeded: unknown', $code));
80
+ }
81
+ throw new liveagent_Exception_SettingNotValid($message);
82
+ }*/
83
+ }
84
+
85
+ public function getOwnerSessionId() {
86
+ return $this->getOption(Qualityunit_Liveagent_Helper_Settings::OWNER_SESSIONID);
87
+ }
88
+
89
+ public function getOwnerAuthToken() {
90
+ $token = $this->getOption(Qualityunit_Liveagent_Helper_Settings::OWNER_AUTHTOKEN);
91
+ if ($token == '') {
92
+ $this->login();
93
+ $token = $this->getOption(Qualityunit_Liveagent_Helper_Settings::OWNER_AUTHTOKEN);
94
+ }
95
+ return $token;
96
+ }
97
+
98
+ public function login() {
99
+ $auth = new Qualityunit_Liveagent_Helper_Auth();
100
+ $loginData = $auth->LoginAndGetLoginData();
101
+ try {
102
+ $sessionId = $loginData->getValue('session');
103
+ $this->setOption(Qualityunit_Liveagent_Helper_Settings::OWNER_SESSIONID, $sessionId);
104
+ } catch (La_Data_RecordSetNoRowException $e) {
105
+ throw new Qualityunit_Liveagent_Exception_ConnectProblem();
106
+ }
107
+ try {
108
+ $token = $sessionId = $loginData->getValue('authtoken');
109
+ $this->setOption(Qualityunit_Liveagent_Helper_Settings::OWNER_AUTHTOKEN, $token);
110
+ } catch (La_Data_RecordSetNoRowException $e) {
111
+ // we are communicating with older LA that does not send auth token
112
+ $this->setOption(Qualityunit_Liveagent_Helper_Settings::OWNER_AUTHTOKEN, '');
113
+ }
114
+ return $sessionId;
115
+ }
116
+
117
+ public function settingsDefinedForConnection() {
118
+ //return strlen(trim($this->getLiveAgentUrl())) && strlen(trim($this->getOwnerEmail()));
119
+ }
120
+
121
+ public function getButtonsGridRecordset() {
122
+ /*try {
123
+ $data = unserialize($this->getCachedSetting(self::BUTTONS_DATA));
124
+ } catch (liveagent_Exception_SettingNotValid $e) {
125
+ $buttonsHelper = new liveagent_helper_Buttons();
126
+ $data = $buttonsHelper->getData();
127
+ if ($data->getSize() == 0) {
128
+ return $data;
129
+ }
130
+ $this->setCachedSetting(self::BUTTONS_DATA, serialize($data));
131
+ }
132
+ return $data;*/
133
+ }
134
+
135
+ public function getLiveAgentUrl() {
136
+ return $this->getOption(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME);
137
+ }
138
+
139
+ public function getOwnerEmail() {
140
+ return $this->getOption(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME);
141
+ //return get_option(self::LA_OWNER_EMAIL_SETTING_NAME);
142
+ }
143
+
144
+ public function getOwnerPassword() {
145
+ return $this->getOption(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_PASSWORD_SETTING_NAME);
146
+ }
147
+
148
+ public function buttonIsEnabled($buttonId) {
149
+ /*$value = get_option(liveagent_Settings::BUTTONS_CONFIGURATION_SETTING_NAME);
150
+ if ($value == '' || $value === null) {
151
+ return false;
152
+ }
153
+ if (array_key_exists($buttonId, $value) && $value[$buttonId] == 'true') {
154
+ return true;
155
+ }
156
+ return false;*/
157
+ }
158
+
159
+ public function getButtonsLastReloadTime() {
160
+ return $this->getOption(self::LAST_BUTTONS_RELOAD_TIME);
161
+ }
162
+
163
+ public function updateLastButtonsReloadTime() {
164
+ $this->setOption(self::LAST_BUTTONS_RELOAD_TIME, time());
165
+ }
166
+
167
+ public function setOption($name, $value) {
168
+ $setting = Mage::getModel('liveagent/settings')->load($name, 'name');
169
+ $setting->setValue($value);
170
+ $setting->setName($name);
171
+ $setting->save();
172
+ }
173
+
174
+ public function getOption($name) {
175
+ $setting = Mage::getModel('liveagent/settings')->load($name, 'name');
176
+ return $setting->getData('value');
177
+ }
178
+
179
+ public function getAccountStatus() {
180
+ return $this->getOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_STATUS);
181
+ }
182
+
183
+ public function connectionSettingsAreEmpty() {
184
+ return $this->getOption(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME) == '' && $this->getOption(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME) == '';
185
+ }
186
+ }
187
+ ?>
app/code/local/Qualityunit/Liveagent/Helper/Signup.php ADDED
@@ -0,0 +1,26 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * @copyright Copyright (c) 2007 Quality Unit s.r.o.
4
+ * @author Juraj Simon
5
+ * @package WpLiveAgentPlugin
6
+ * @version 1.0.0
7
+ *
8
+ * Licensed under GPL2
9
+ */
10
+
11
+ class Qualityunit_Liveagent_Helper_Signup extends Qualityunit_Liveagent_Helper_Base {
12
+
13
+ public function signup($name, $email, $domain, $password, $papVisitorId = '') {
14
+ $request = new La_Rpc_ActionRequest('Dp_QualityUnit_La_Signup', 'createAccountRequest');
15
+ $request->setUrl('http://members.qualityunit.com/scripts/server.php');
16
+ $request->addParam('domain', $domain);
17
+ $request->addParam('name', $name);
18
+ $request->addParam('email', $email);
19
+ $request->addParam('password', $password);
20
+ $request->addParam('variationid', '00198b52');
21
+ $request->addParam('PAPvisitorId', $papVisitorId);
22
+ $request->addParam('source', 'magento');
23
+ $request->sendNow();
24
+ return $request->getStdResponse();
25
+ }
26
+ }
app/code/local/Qualityunit/Liveagent/Model/Buttons.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Buttons extends Mage_Core_Model_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ parent::_construct();
8
+ $this->_init('liveagent/buttons');
9
+ }
10
+ }
app/code/local/Qualityunit/Liveagent/Model/Liveagent.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Liveagent extends Mage_Core_Model_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ parent::_construct();
8
+ $this->_init('liveagent/liveagent');
9
+ }
10
+ }
app/code/local/Qualityunit/Liveagent/Model/Mysql4/Buttons.php ADDED
@@ -0,0 +1,9 @@
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Mysql4_Buttons extends Mage_Core_Model_Mysql4_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ $this->_init('liveagent/liveagentbuttons', 'liveagentbutton_id');
8
+ }
9
+ }
app/code/local/Qualityunit/Liveagent/Model/Mysql4/Buttons/Collection.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Mysql4_Buttons_Collection extends Mage_Core_Model_Mysql4_Collection_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ parent::_construct();
8
+ $this->_init('liveagent/buttons');
9
+ }
10
+ }
app/code/local/Qualityunit/Liveagent/Model/Mysql4/Liveagent.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Mysql4_Liveagent extends Mage_Core_Model_Mysql4_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ // Note that the liveagent_id refers to the key field in your database table.
8
+ $this->_init('liveagent/liveagent', 'liveagent_id');
9
+ }
10
+ }
app/code/local/Qualityunit/Liveagent/Model/Mysql4/Liveagent/Collection.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Mysql4_Liveagent_Collection extends Mage_Core_Model_Mysql4_Collection_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ parent::_construct();
8
+ $this->_init('liveagent/liveagent');
9
+ }
10
+ }
app/code/local/Qualityunit/Liveagent/Model/Mysql4/Settings.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Mysql4_Settings extends Mage_Core_Model_Mysql4_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ // Note that the liveagent_id refers to the key field in your database table.
8
+ $this->_init('liveagent/liveagentsettings', 'id');
9
+ }
10
+ }
app/code/local/Qualityunit/Liveagent/Model/Mysql4/Settings/Collection.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Mysql4_Settings_Collection extends Mage_Core_Model_Mysql4_Collection_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ parent::_construct();
8
+ $this->_init('liveagent/liveagentsettings');
9
+ }
10
+ }
app/code/local/Qualityunit/Liveagent/Model/Settings.php ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Settings extends Mage_Core_Model_Abstract
4
+ {
5
+ public function _construct()
6
+ {
7
+ parent::_construct();
8
+ $this->_init('liveagent/settings');
9
+ }
10
+ }
app/code/local/Qualityunit/Liveagent/Model/Status.php ADDED
@@ -0,0 +1,15 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Model_Status extends Varien_Object
4
+ {
5
+ const STATUS_ENABLED = 'Y';
6
+ const STATUS_DISABLED = 'N';
7
+
8
+ static public function getOptionArray()
9
+ {
10
+ return array(
11
+ self::STATUS_ENABLED => Mage::helper('liveagent')->__('Enabled'),
12
+ self::STATUS_DISABLED => Mage::helper('liveagent')->__('Disabled')
13
+ );
14
+ }
15
+ }
app/code/local/Qualityunit/Liveagent/controllers/Adminhtml/ButtonsController.php ADDED
@@ -0,0 +1,84 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Adminhtml_ButtonsController extends Mage_Adminhtml_Controller_action {
4
+
5
+ /**
6
+ * @var Qualityunit_Liveagent_Helper_Settings
7
+ **/
8
+ private $settings = null;
9
+
10
+ protected function _construct() {
11
+ parent::_construct();
12
+ $path = dirname(__FILE__);
13
+ require_once $path . '/../../Helper/PhpApi.php';
14
+ $this->settings = new Qualityunit_Liveagent_Helper_Settings();
15
+ }
16
+
17
+ private function initLayout() {
18
+ $this->loadLayout()
19
+ ->_setActiveMenu('liveagent/buttons')
20
+ ->_addBreadcrumb(Mage::helper('adminhtml')->__('Buttons'), Mage::helper('adminhtml')->__('Settings'));
21
+ }
22
+
23
+ private function doAfterPost() {
24
+ $this->_redirect('*/*');
25
+ }
26
+
27
+ private function doPostAction() {
28
+ $post = $this->getRequest()->getPost();
29
+ $this->doAfterPost();
30
+ }
31
+
32
+ public function postAction() {
33
+ $this->initLayout();
34
+ try {
35
+ $this->doPostAction();
36
+ } catch (Exception $e) {
37
+ Mage::log($e->getMessage(), Zend_log::ERR);
38
+ Mage::getSingleton('adminhtml/session')->addError($e->getMessage());
39
+ $this->_redirect('*/*');
40
+ $this->renderLayout();
41
+ }
42
+ }
43
+
44
+ public function indexAction() {
45
+ $this->initLayout();
46
+ try {
47
+ $this->doIndexAction();
48
+ } catch (Exception $e) {
49
+ Mage::log($e->getMessage(), Zend_log::ERR);
50
+ Mage::getSingleton('adminhtml/session')->addError($e->getMessage());
51
+ $this->renderLayout();
52
+ }
53
+ }
54
+
55
+ public function doIndexAction() {
56
+ $block = new Qualityunit_Liveagent_Block_ButtonsGrid();
57
+ $this->_addContent($this->getLayout()->createBlock($block));
58
+ $this->renderLayout();
59
+ }
60
+
61
+ public function massStatusAction()
62
+ {
63
+ $liveagentIds = $this->getRequest()->getParam('buttonids');
64
+ if(!is_array($liveagentIds)) {
65
+ Mage::getSingleton('adminhtml/session')->addError($this->__('Please select item(s)'));
66
+ } else {
67
+ try {
68
+ foreach ($liveagentIds as $liveagentId) {
69
+ $liveagent = Mage::getSingleton('liveagent/buttons')
70
+ ->load($liveagentId)
71
+ ->setOnlinestatus($this->getRequest()->getParam('status'))
72
+ ->setIsMassupdate(true)
73
+ ->save();
74
+ }
75
+ $this->_getSession()->addSuccess(
76
+ $this->__('Total of %d button(s) were successfully updated', count($liveagentIds))
77
+ );
78
+ } catch (Exception $e) {
79
+ $this->_getSession()->addError($e->getMessage());
80
+ }
81
+ }
82
+ $this->_redirect('*/*/index');
83
+ }
84
+ }
app/code/local/Qualityunit/Liveagent/controllers/Adminhtml/LiveagentController.php ADDED
@@ -0,0 +1,242 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ class Qualityunit_Liveagent_Adminhtml_LiveagentController extends Mage_Adminhtml_Controller_Action {
4
+
5
+ const LIVEAGENT_PLUGIN_NAME = 'liveagent';
6
+ const ACTION_CREATE = 'c';
7
+
8
+ /**
9
+ * @var Qualityunit_Liveagent_Helper_Settings
10
+ **/
11
+ private $settings = null;
12
+
13
+ protected function _construct() {
14
+ parent::_construct();
15
+ $path = dirname(__FILE__);
16
+ require_once $path . '/../../Helper/PhpApi.php';
17
+ $this->settings = new Qualityunit_Liveagent_Helper_Settings();
18
+ }
19
+
20
+ private function initLayout() {
21
+ $this->loadLayout()
22
+ ->_setActiveMenu('liveagent/account')
23
+ ->_addBreadcrumb(Mage::helper('adminhtml')->__('Account'), Mage::helper('adminhtml')->__('Settings'));
24
+ }
25
+
26
+ private function doAfterPost() {
27
+ $this->_redirect('*/*');
28
+ }
29
+
30
+ private function doPostAction() {
31
+ $post = $this->getRequest()->getPost();
32
+ if (!array_key_exists('action', $post)) {
33
+ $this->doAfterPost();
34
+ return;
35
+ }
36
+ if ($post['action']=="r") {
37
+ try {
38
+ $this->validateSignupMail($post[Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME]);
39
+ $this->registerAccountAction(
40
+ $post[Qualityunit_Liveagent_Block_Signup::FULL_NAME_FIELD],
41
+ $post[Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME],
42
+ $post[Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME],
43
+ substr(md5(microtime()),0,8),
44
+ ''
45
+ );
46
+ } catch (Qualityunit_Liveagent_Exception_SignupFailed $e) {
47
+ $this->signupFailed($e);
48
+ return;
49
+ }
50
+ $this->doAfterPost();
51
+ return;
52
+ }
53
+ if ($post['action']==Qualityunit_Liveagent_Block_Account::SAVE_ACCOUNT_SETTINGS_ACTION_FLAG) {
54
+ $this->saveAccountSettings($post);
55
+ Mage::getSingleton('adminhtml/session')->addSuccess(Mage::helper('adminhtml')->__('Account settings were saved successfully'));
56
+ }
57
+ $this->doAfterPost();
58
+ }
59
+
60
+ private function validateSignupMail($mail) {
61
+ if (!Zend_Validate::is($mail, 'EmailAddress')) {
62
+ throw new Qualityunit_Liveagent_Exception_SignupFailed(Mage::helper('adminhtml')->__('Please enter valid email address'));
63
+ }
64
+ }
65
+
66
+ private function signupFailed($e) {
67
+ Mage::getSingleton('adminhtml/session')->addError($e->getMessage());
68
+ $this->doAfterPost();
69
+ }
70
+
71
+ public function postAction() {
72
+ $this->initLayout();
73
+ try {
74
+ $this->doPostAction();
75
+ } catch (Exception $e) {
76
+ Mage::log($e->getMessage(), Zend_log::ERR);
77
+ Mage::getSingleton('adminhtml/session')->addError($e->getMessage());
78
+ $this->_redirect('*/*');
79
+ $this->renderLayout();
80
+ }
81
+ }
82
+
83
+ private function saveAccountsettings($post) {
84
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME, $post[Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME]);
85
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_PASSWORD_SETTING_NAME, $post[Qualityunit_Liveagent_Helper_Settings::LA_OWNER_PASSWORD_SETTING_NAME]);
86
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME, $post[Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME]);
87
+ }
88
+
89
+ public function indexAction() {
90
+ $this->initLayout();
91
+ try {
92
+ $this->doIndexAction();
93
+ } catch (Exception $e) {
94
+ Mage::log($e->getMessage(), Zend_log::ERR);
95
+ Mage::getSingleton('adminhtml/session')->addError($e->getMessage());
96
+ $this->renderLayout();
97
+ }
98
+ }
99
+
100
+ private function getAccountNotReachableTimes() {
101
+ $times = $this->settings->getOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_NOT_REACHABLE_TIMES);
102
+ if ($times == '') {
103
+ return 0;
104
+ }
105
+ return $times;
106
+ }
107
+
108
+ /**
109
+ * @return Qualityunit_Liveagent_Block_Waiting
110
+ */
111
+ private function getWaitingDialog() {
112
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_NOT_REACHABLE_TIMES, $this->settings->getOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_NOT_REACHABLE_TIMES) + 1);
113
+ return new Qualityunit_Liveagent_Block_Waiting();
114
+ }
115
+
116
+ public function doIndexAction() {
117
+ if ($this->getRequest()->getParam('action')=="") {
118
+ if ($this->isNew()) {
119
+ $block = new Qualityunit_Liveagent_Block_Signup();
120
+ } else if ($this->isWaiting()) {
121
+ if ($this->getAccountNotReachableTimes() <= 1) {
122
+ $block = $this->getWaitingDialog();
123
+ } else if ($this->getAccountNotReachableTimes() > 1) {
124
+ if ($this->accountIsOnline() || $this->getAccountNotReachableTimes() > 3) {
125
+ $this->skipWaitAction();
126
+ $this->renderAccountDialog();
127
+ $this->renderLayout();
128
+ return;
129
+ }
130
+ $block = $this->getWaitingDialog();
131
+ }
132
+ } else {
133
+ $this->renderAccountDialog();
134
+ $this->renderLayout();
135
+ return;
136
+ }
137
+ $this->_addContent($this->getLayout()->createBlock($block));
138
+ $this->renderLayout();
139
+ return;
140
+ }
141
+ if ($this->getRequest()->getParam('action')=="c") {
142
+ $this->resetAccountAction();
143
+ $block = new Qualityunit_Liveagent_Block_Signup();
144
+ }
145
+ if ($this->getRequest()->getParam('action')=="s") {
146
+ $this->skipWaitAction();
147
+ $this->renderAccountDialog();
148
+ $this->renderLayout();
149
+ return;
150
+ }
151
+ $this->_addContent($this->getLayout()->createBlock($block));
152
+ $this->renderLayout();
153
+ }
154
+
155
+ private function renderConnectionInfo() {
156
+ $block = new Qualityunit_Liveagent_Block_Connectioninfo();
157
+ $block->setConnectionValid($this->accountIsOnline());
158
+ $this->_addContent($this->getLayout()->createBlock($block));
159
+ }
160
+
161
+ private function renderAccountButton() {
162
+ $block = new Qualityunit_Liveagent_Block_Accountbutton();
163
+ $block->setSettings($this->settings);
164
+ $this->_addContent($this->getLayout()->createBlock($block));
165
+ }
166
+
167
+ private function renderCreateFreeAccountWidget() {
168
+ $block = new Qualityunit_Liveagent_Block_CreateFreeAccountWidget();
169
+ $this->_addContent($this->getLayout()->createBlock($block));
170
+ }
171
+
172
+ private function renderAccountDialog() {
173
+ $this->renderConnectionInfo();
174
+ if ($this->accountIsOnline()) {
175
+ $buttons = new Qualityunit_Liveagent_Helper_Buttons();
176
+ $buttons->refresh();
177
+ $buttons->enableDefaultButtonIfNoImpressions();
178
+ $this->renderAccountButton();
179
+ } else {
180
+ if ($this->settings->connectionSettingsAreEmpty()) {
181
+ $this->renderCreateFreeAccountWidget();
182
+ }
183
+ }
184
+ $block = new Qualityunit_Liveagent_Block_Account();
185
+ $this->_addContent($this->getLayout()->createBlock($block));
186
+ }
187
+
188
+ private function accountIsOnline() {
189
+ $auth = new Qualityunit_Liveagent_Helper_Auth();
190
+ try {
191
+ $auth->ping();
192
+ Mage::log('Account is online!', Zend_Log::INFO);
193
+ return true;
194
+ } catch (Qualityunit_Liveagent_Exception_ConnectProblem $e) {
195
+ Mage::log('Account not online', Zend_Log::INFO);
196
+ return false;
197
+ }
198
+ }
199
+
200
+ private function isNew() {
201
+ return $this->settings->getAccountStatus() == Qualityunit_Liveagent_Helper_Base::ACCOUNT_STATUS_NOTSET;
202
+ }
203
+
204
+ private function isWaiting() {
205
+ return $this->settings->getAccountStatus() == Qualityunit_Liveagent_Helper_Base::ACCOUNT_STATUS_WAIT;
206
+ }
207
+
208
+ private function resetAccountAction() {
209
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_STATUS, Qualityunit_Liveagent_Helper_Base::ACCOUNT_STATUS_NOTSET);
210
+ }
211
+
212
+ private function sendSignupRequest($name, $email, $domain, $password, $papvisitorId) {
213
+ $signupHelper = new Qualityunit_Liveagent_Helper_Signup();
214
+ try {
215
+ $response = $signupHelper->signup($name, $email, $domain, $password, $papvisitorId);
216
+ } catch (La_Exception $e) {
217
+ throw new Qualityunit_Liveagent_Exception_SignupFailed($e->getMessage());
218
+ }
219
+ Mage::log("Signup response recieved:" . print_r($response, true), Zend_log::DEBUG);
220
+ if ($response->success != "Y") {
221
+ Mage::log("Response contain error:" . $response->errorMessage, Zend_log::DEBUG);
222
+ throw new Qualityunit_Liveagent_Exception_SignupFailed($response->errorMessage);
223
+ }
224
+ Mage::log("Response OK", Zend_log::DEBUG);
225
+ }
226
+
227
+ private function registerAccountAction($name, $email, $domain, $password, $papvisitorId) {
228
+ $this->sendSignupRequest($name, $email, $domain, $password, $papvisitorId);
229
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::LA_URL_SETTING_NAME, 'http://' . $domain . '.ladesk.com');
230
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_EMAIL_SETTING_NAME, $email);
231
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::LA_OWNER_PASSWORD_SETTING_NAME, $password);
232
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_STATUS, Qualityunit_Liveagent_Helper_Base::ACCOUNT_STATUS_WAIT);
233
+ }
234
+
235
+ private function skipWaitAction() {
236
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_STATUS, Qualityunit_Liveagent_Helper_Base::ACCOUNT_STATUS_SET);
237
+ $this->settings->setOption(Qualityunit_Liveagent_Helper_Settings::ACCOUNT_NOT_REACHABLE_TIMES, 0);
238
+ $buttons = new Qualityunit_Liveagent_Helper_Buttons();
239
+ $buttons->enableDefaultButton();
240
+ $buttons->enableDefaultButton('button1');
241
+ }
242
+ }
app/code/local/Qualityunit/Liveagent/controllers/IndexController.php ADDED
@@ -0,0 +1,9 @@
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ class Qualityunit_Liveagent_IndexController extends Mage_Core_Controller_Front_Action
3
+ {
4
+ public function indexAction()
5
+ {
6
+ $this->loadLayout();
7
+ $this->renderLayout();
8
+ }
9
+ }
app/code/local/Qualityunit/Liveagent/etc/config.xml ADDED
@@ -0,0 +1,153 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <config>
3
+ <modules>
4
+ <Qualityunit_Liveagent>
5
+ <version>0.1.0</version>
6
+ </Qualityunit_Liveagent>
7
+ </modules>
8
+ <frontend>
9
+ <routers>
10
+ <liveagent>
11
+ <use>standard</use>
12
+ <args>
13
+ <module>Qualityunit_Liveagent</module>
14
+ <frontName>liveagent</frontName>
15
+ </args>
16
+ </liveagent>
17
+ </routers>
18
+ <layout>
19
+ <updates>
20
+ <liveagent>
21
+ <file>liveagent.xml</file>
22
+ </liveagent>
23
+ </updates>
24
+ </layout>
25
+ </frontend>
26
+ <admin>
27
+ <routers>
28
+ <liveagent>
29
+ <use>admin</use>
30
+ <args>
31
+ <module>Qualityunit_Liveagent</module>
32
+ <frontName>liveagent</frontName>
33
+ </args>
34
+ </liveagent>
35
+ </routers>
36
+ </admin>
37
+ <adminhtml>
38
+ <menu>
39
+ <liveagent module="liveagent">
40
+ <title>Liveagent</title>
41
+ <sort_order>71</sort_order>
42
+ <children>
43
+ <account module="liveagent">
44
+ <title>Account settings</title>
45
+ <sort_order>0</sort_order>
46
+ <action>liveagent/adminhtml_liveagent</action>
47
+ </account>
48
+ <buttons module="liveagent">
49
+ <title>Buttons</title>
50
+ <sort_order>1</sort_order>
51
+ <action>liveagent/adminhtml_buttons</action>
52
+ </buttons>
53
+ </children>
54
+ </liveagent>
55
+ </menu>
56
+ <acl>
57
+ <resources>
58
+ <all>
59
+ <title>Allow Everything</title>
60
+ </all>
61
+ <admin>
62
+ <children>
63
+ <Qualityunit_Liveagent>
64
+ <title>Liveagent Module</title>
65
+ <sort_order>10</sort_order>
66
+ </Qualityunit_Liveagent>
67
+ </children>
68
+ </admin>
69
+ </resources>
70
+ </acl>
71
+ <layout>
72
+ <updates>
73
+ <liveagent>
74
+ <file>liveagent.xml</file>
75
+ </liveagent>
76
+ </updates>
77
+ </layout>
78
+ </adminhtml>
79
+ <global>
80
+ <models>
81
+ <liveagent>
82
+ <class>Qualityunit_Liveagent_Model</class>
83
+ <resourceModel>liveagent_mysql4</resourceModel>
84
+ </liveagent>
85
+ <liveagent_mysql4>
86
+ <class>Qualityunit_Liveagent_Model_Mysql4</class>
87
+ <entities>
88
+ <liveagent>
89
+ <table>liveagent</table>
90
+ </liveagent>
91
+ <liveagentsettings>
92
+ <table>liveagentsettings</table>
93
+ </liveagentsettings>
94
+ <liveagentbuttons>
95
+ <table>liveagentbuttons</table>
96
+ </liveagentbuttons>
97
+ </entities>
98
+ </liveagent_mysql4>
99
+ </models>
100
+ <resources>
101
+ <liveagent_setup>
102
+ <setup>
103
+ <module>Qualityunit_Liveagent</module>
104
+ </setup>
105
+ <connection>
106
+ <use>core_setup</use>
107
+ </connection>
108
+ </liveagent_setup>
109
+ <liveagent_write>
110
+ <connection>
111
+ <use>core_write</use>
112
+ </connection>
113
+ </liveagent_write>
114
+ <liveagent_read>
115
+ <connection>
116
+ <use>core_read</use>
117
+ </connection>
118
+ </liveagent_read>
119
+ </resources>
120
+ <blocks>
121
+ <liveagent>
122
+ <class>Qualityunit_Liveagent_Block</class>
123
+ </liveagent>
124
+ <dialog>
125
+ <class>Qualityunit_Liveagent_Block</class>
126
+ </dialog>
127
+ <connectioninfo>
128
+ <class>Qualityunit_Liveagent_Block</class>
129
+ </connectioninfo>
130
+ <accountbutton>
131
+ <class>Qualityunit_Liveagent_Block</class>
132
+ </accountbutton>
133
+ <page>
134
+ <rewrite>
135
+ <html_head>Qualityunit_Liveagent_Block_CustomHeader</html_head>
136
+ </rewrite>
137
+ </page>
138
+ <adminhtml>
139
+ <rewrite>
140
+ <page_head>Qualityunit_Liveagent_Block_CustomHeader2</page_head>
141
+ </rewrite>
142
+ </adminhtml>
143
+ </blocks>
144
+ <helpers>
145
+ <liveagent>
146
+ <class>Qualityunit_Liveagent_Helper</class>
147
+ </liveagent>
148
+ <buttons>
149
+ <class>Qualityunit_Liveagent_Helper</class>
150
+ </buttons>
151
+ </helpers>
152
+ </global>
153
+ </config>
app/code/local/Qualityunit/Liveagent/sql/liveagent_setup/mysql4-install-0.1.0.php ADDED
@@ -0,0 +1,59 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+
3
+ $installer = $this;
4
+
5
+ $installer->startSetup();
6
+
7
+ /*$installer->run("
8
+
9
+ -- DROP TABLE IF EXISTS {$this->getTable('liveagent')};
10
+ CREATE TABLE {$this->getTable('liveagent')} (
11
+ `liveagent_id` int(11) unsigned NOT NULL auto_increment,
12
+ `title` varchar(255) NOT NULL default '',
13
+ `filename` varchar(255) NOT NULL default '',
14
+ `content` text NOT NULL default '',
15
+ `status` smallint(6) NOT NULL default '0',
16
+ `created_time` datetime NULL,
17
+ `update_time` datetime NULL,
18
+ PRIMARY KEY (`liveagent_id`)
19
+ ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
20
+
21
+ ");*/
22
+
23
+ $installer->run("
24
+ DROP TABLE IF EXISTS {$this->getTable('liveagentbuttons')};
25
+ CREATE TABLE {$this->getTable('liveagentbuttons')} (
26
+ `liveagentbutton_id` int(11) unsigned NOT NULL auto_increment,
27
+ `buttonid` char(8) NOT NULL,
28
+ `userid` char(8) DEFAULT NULL,
29
+ `name` text,
30
+ `contenttype` char(1) DEFAULT NULL,
31
+ `onlinestatus` char(1) DEFAULT NULL,
32
+ `onlinecode` longtext,
33
+ `offlinecode` longtext,
34
+ `window_width` varchar(10) DEFAULT NULL,
35
+ `window_height` varchar(10) DEFAULT NULL,
36
+ `window_position` varchar(20) DEFAULT NULL,
37
+ `impressions` int(11) DEFAULT '0',
38
+ `chats` int(11) DEFAULT '0',
39
+ `data1` varchar(255) DEFAULT NULL,
40
+ `data2` varchar(255) DEFAULT NULL,
41
+ `data3` varchar(255) DEFAULT NULL,
42
+ `data4` varchar(255) DEFAULT NULL,
43
+ `data5` varchar(255) DEFAULT NULL,
44
+ PRIMARY KEY (`liveagentbutton_id`)
45
+ ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
46
+
47
+ DROP TABLE IF EXISTS {$this->getTable('liveagentsettings')};
48
+ CREATE TABLE {$this->getTable('liveagentsettings')} (
49
+ `id` int(11) unsigned NOT NULL auto_increment,
50
+ `name` varchar(255) NOT NULL,
51
+ `value` varchar(255) DEFAULT NULL,
52
+ PRIMARY KEY (`id`)
53
+ ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
54
+
55
+ INSERT INTO liveagentsettings (`name` ,`value`) VALUES ('la-settings_accountstatus', 'N');
56
+ INSERT INTO liveagentsettings (`name` ,`value`) VALUES ('la-settings_accountnotreachabletimes', '0');
57
+ ");
58
+
59
+ $installer->endSetup();
app/design/adminhtml/default/default/layout/liveagent.xml ADDED
@@ -0,0 +1,8 @@
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <layout version="0.1.0">
3
+ <liveagent_adminhtml_liveagent_index>
4
+ <reference name="content">
5
+ <block type="liveagent/dialog" name="dialog" template="liveagent/dialog.phtml"/>
6
+ </reference>
7
+ </liveagent_adminhtml_liveagent_index>
8
+ </layout>
app/design/adminhtml/default/default/template/liveagent/dialog.phtml ADDED
@@ -0,0 +1 @@
 
1
+ <!-- dialog content template-->
app/design/frontend/base/default/layout/liveagent.xml ADDED
@@ -0,0 +1,11 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <layout version="0.1.0">
3
+ <default>
4
+ <reference name="head">
5
+ <block type="core/template" name="liveagent_header" template="liveagent/header.phtml" />
6
+ </reference>
7
+ <reference name="after_body_start">
8
+ <block type="core/template" name="liveagent_button" template="liveagent/button.phtml" />
9
+ </reference>
10
+ </default>
11
+ </layout>
app/design/frontend/base/default/template/liveagent/button.phtml ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ $collection = Mage::getModel('liveagent/buttons')->getCollection()->addFilter('contenttype', 'F')->addFilter('onlinestatus', 'Y');
3
+ $setting = Mage::getModel('liveagent/settings')->load('la-url', 'name');
4
+ $url = $setting->getValue();
5
+ echo '<div id="la_x2s6df8d"/>';
6
+ foreach ($collection as $button) {
7
+ echo '<img src="'.$url . '/scripts/pix.gif" onLoad="LiveAgentTracker.createButton(\''.$button->getButtonid().'\', this);"/>' . "\n";
8
+ }
9
+
10
+ ?>
app/design/frontend/base/default/template/liveagent/header.phtml ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
1
+ <?php
2
+ $setting = Mage::getModel('liveagent/settings')->load('la-url', 'name');
3
+ $url = $setting->getValue();
4
+ echo '<script type="text/javascript" id="la_x2s6df8d" src="'.$url.'/scripts/trackjs.php"></script>';
5
+ ?>
app/etc/modules/Qualityunit_Liveagent.xml ADDED
@@ -0,0 +1,9 @@
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <config>
3
+ <modules>
4
+ <Qualityunit_Liveagent>
5
+ <active>true</active>
6
+ <codePool>local</codePool>
7
+ </Qualityunit_Liveagent>
8
+ </modules>
9
+ </config>
package.xml ADDED
@@ -0,0 +1,20 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?xml version="1.0"?>
2
+ <package>
3
+ <name>Liveagent</name>
4
+ <version>1.0.15</version>
5
+ <stability>stable</stability>
6
+ <license uri="http://www.gnu.org/licenses/gpl-3.0.txt">GPLv3</license>
7
+ <channel>community</channel>
8
+ <extends/>
9
+ <summary>LiveAgent plugin integrates well known help desk and live chat software into any Magento installation. No HTML knowledge is required.</summary>
10
+ <description>Magento-LiveAgent plugin integrates full featured help desk and live chat software Live Agent into any Magento installation.&#xD;
11
+ Simply add "start chat" button by few simple clicks and be live within 5 minutes.</description>
12
+ <notes>changes:&#xD;
13
+ - removes magento compilator errors</notes>
14
+ <authors><author><name>Juraj Simon</name><user>jurajsim</user><email>jsimon@qualityunit.com</email></author></authors>
15
+ <date>2012-01-25</date>
16
+ <time>10:48:29</time>
17
+ <contents><target name="magelocal"><dir name="Qualityunit"><dir name="Liveagent"><dir name="Block"><file name="Account.php" hash="3c5780c7e43a1b427c8a9942be5ae5f4"/><file name="Accountbutton.php" hash="ed893391cef66ed9a20270dcd825e8a7"/><dir name="Adminhtml"><dir name="Config"><file name="Account.php" hash="bd7cfa5110511019f7a163a3e1237f8d"/></dir><dir name="Liveagent"><file name="Grid.php" hash="bd7cfa5110511019f7a163a3e1237f8d"/></dir><file name="Liveagent.php" hash="39a36de680b8bd0f892cac2eceda9aa3"/></dir><file name="Base.php" hash="be036bcec33c8cc8b6248f04e66ab2ae"/><file name="ButtonsGrid.php" hash="403f095a11c3cbdd4c8d2bdd5c39ea92"/><file name="Connectioninfo.php" hash="0ba2988926c3a458b402030f90a9c6c8"/><file name="CreateFreeAccountWidget.php" hash="69673bbf0662db1ed97d5055184707d8"/><file name="CustomHeader.php" hash="d2f34ee2a9338729d6e72c328469eb2f"/><file name="CustomHeader2.php" hash="bcab9079100055561d93f38e35eefcad"/><file name="Dialog.php" hash="b599ad757d352cd8205f5eb297b33aaf"/><file name="ErrorMessageWidget.php" hash="c2464b0cc51dcc66c2a1da4f5d7196ce"/><file name="FloatButtonWidget.php" hash="e7541c92e18c59dce02435de291ab117"/><file name="Liveagent.php" hash="b2a3acbd39f442b2c1f41844a93c1fed"/><file name="Signup.php" hash="519d3b0126b7add1dd6f9f17fb956457"/><file name="Waiting.php" hash="5d305d888e7268885f1df43976b6d67a"/></dir><dir name="Exception"><file name="ConnectProblem.php" hash="f2c9840d533e6389ba84989337de32c9"/><file name="SignupFailed.php" hash="e370773d3b306bb8c24e5263d1491860"/></dir><dir name="Helper"><file name="Auth.php" hash="8263ab6dbff508b343bb2dda38227f3d"/><file name="Base.php" hash="8003134735200ef67da5e32810373dca"/><file name="Buttons.php" hash="050bef921fc945e18b21fc03120610b1"/><file name="Data.php" hash="e390bbfd2177732fb4701692dcdf9e21"/><file name="PhpApi.php" hash="be07541b661295f7d587355719f90641"/><file name="Settings.php" hash="1ad5a2e8e0380e25761ef560e4df0f86"/><file name="Signup.php" hash="54ee655169693c1434ba57d19b4ba236"/></dir><dir name="Model"><file name="Buttons.php" hash="ab4d53d1fdf7266f086f56fe5b789d9a"/><file name="Liveagent.php" hash="8bbff1b87bc9359b20fd57e903cc36a7"/><dir name="Mysql4"><dir name="Buttons"><file name="Collection.php" hash="1aa2f16ba17d7b35622aac90e49ab0d8"/></dir><file name="Buttons.php" hash="0b8cdf68a1a733ac9c0b2306bb7d707d"/><dir name="Liveagent"><file name="Collection.php" hash="4d97914f2b313bf94c153825926a24c6"/></dir><file name="Liveagent.php" hash="ce9e8956b01d6adbd8f6e47d90a940db"/><dir name="Settings"><file name="Collection.php" hash="90b3bf0169c4954eca8ad1fafc727cdf"/></dir><file name="Settings.php" hash="742ed68fe5865687b4d3663274720c80"/></dir><file name="Settings.php" hash="295eda087e5e7e3a7125d57823a0059c"/><file name="Status.php" hash="7c974f2495a419f687ce0f13421b4e59"/></dir><dir name="controllers"><dir name="Adminhtml"><file name="ButtonsController.php" hash="a6beefa9175c7eb5fcb5463eabecb035"/><file name="LiveagentController.php" hash="9a15220aa00775d759f147c4a922edbc"/></dir><file name="IndexController.php" hash="6ed77d2da66be08bd439f68820dbf1c0"/></dir><dir name="etc"><file name="config.xml" hash="048a093bb26aee02c3a441f7cf9ef131"/></dir><dir name="sql"><dir name="liveagent_setup"><file name="mysql4-install-0.1.0.php" hash="c3b21fc18c4f4e51b118670ecb50952f"/></dir></dir></dir><file name=".buildpath" hash="0d2475e3be03419439e2a22ebc4e0813"/><file name=".project" hash="41fa1353a5ab19352d5636d52e17351d"/><dir name=".settings"><file name="org.eclipse.php.core.prefs" hash="abf8f34f8087365ab830524698638d7c"/><file name="org.eclipse.wst.jsdt.ui.superType.container" hash="b27d1cf62dde4473bab7c433317bb0ce"/><file name="org.eclipse.wst.jsdt.ui.superType.name" hash="c89686a387d2b12b3c729ce35a0bcb5b"/><file name=".jsdtscope" hash="60becf9566a086946234ab57c156ae64"/></dir></dir></target><target name="magedesign"><dir name="adminhtml"><dir name="default"><dir name="default"><dir name="layout"><file name="liveagent.xml" hash="0dcae8c167ac1993cd79ab11c3617726"/></dir><dir name="template"><dir name="liveagent"><file name="dialog.phtml" hash="db91a6f20a5fc16e7f5d492ee8cdc49f"/></dir></dir></dir></dir></dir><dir name="frontend"><dir name="base"><dir name="default"><dir name="layout"><file name="liveagent.xml" hash="8315b710ce2d644aa98666e25e53f24e"/></dir><dir name="template"><dir name="liveagent"><file name="button.phtml" hash="f923d78234b7d814b0f7ac79c89f0900"/><file name="header.phtml" hash="e56cdc01a062e445e0105a6aa13c5999"/></dir></dir></dir></dir></dir></target><target name="mageetc"><dir name="modules"><file name="Qualityunit_Liveagent.xml" hash="db13786c45588e15d5a390201a1709e4"/></dir></target></contents>
18
+ <compatible/>
19
+ <dependencies><required><php><min>5.2.0</min><max>5.6.0</max></php><extension><name>curl</name><min></min><max></max></extension></required></dependencies>
20
+ </package>