Add analytics
[bus.git] / busui / owa / eventQueue.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
<?php 
 
//
// Open Web Analytics - An Open Source Web Analytics Framework
//
// Copyright 2006 Peter Adams. All rights reserved.
//
// Licensed under GPL v2.0 http://www.gnu.org/copyleft/gpl.html
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// $Id$
//
 
if (!class_exists('owa_observer')) {
 
        require_once(OWA_BASE_CLASSES_DIR. 'owa_observer.php');
}
 
if (!class_exists('owa_event') ) {
        require_once(OWA_BASE_CLASS_DIR.'event.php');
}
 
define('OWA_EHS_EVENT_HANDLED', 2);
define('OWA_EHS_EVENT_FAILED', 3);
 
/**
 * Event Dispatcher
 * 
 * @author      Peter Adams <peter@openwebanalytics.com>
 * @copyright   Copyright &copy; 2006 Peter Adams <peter@openwebanalytics.com>
 * @license     http://www.gnu.org/copyleft/gpl.html GPL v2.0
 * @category    owa
 * @package     owa
 * @version             $Revision$            
 * @since               owa 1.0.0
 */
class eventQueue {
        
        /**
         * Stores listeners
         *
         */
        var $listeners = array();
        
        /**
         * Stores listener IDs by event type
         *
         */
        var $listenersByEventType = array();
        
        /**
         * Stores listener IDs by event type
         *
         */
        var $listenersByFilterType = array();
        
        /**
         * Constructor
         *
         */
        function __construct() {
        
        }
        
        /**
         * Attach
         *
         * Attaches observers by event type.
         * Takes a valid user defined callback function for use by PHP's call_user_func_array
         * 
         * @param       $event_name     string
         * @param       $observer       mixed can be a function name or function array
         * @return bool
         */
 
        function attach($event_name, $observer) {
        
        $id = md5(microtime());
        
        // Register event names for this handler
                if(is_array($event_name)) {
                        
                        foreach ($event_name as $k => $name) {  
        
                                $this->listenersByEventType[$name][] = $id;
                        }
                        
                } else {
                
                        $this->listenersByEventType[$event_name][] = $id;       
                }
                
        $this->listeners[$id] = $observer;
               
        return true;
    }
    
    /**
         * Attach
         *
         * Attaches observers by filter type.
         * Takes a valid user defined callback function for use by PHP's call_user_func_array
         * 
         * @param       $filter_name    string
         * @param       $observer       mixed can be a function name or function array
         * @return bool
         */
 
        function attachFilter($filter_name, $observer, $priority = 10) {
        
        $id = md5(microtime());
        
        $this->listenersByFilterType[$filter_name][$priority][] = $id;
                
        $this->listeners[$id] = $observer;
               
    }
 
        /**
         * Notify
         *
         * Notifies all handlers of events in order that they were registered
         * 
         * @param       $event_type     string
         * @param       $event  array
         * @return bool
         */
        function notify($event) {
                
                $responses = array();
                owa_coreAPI::debug("Notifying listeners of ".$event->getEventType());
                //print_r($this->listenersByEventType[$event_type] );
                //print $event->getEventType();
                if (array_key_exists($event->getEventType(), $this->listenersByEventType)) {
                        $list = $this->listenersByEventType[$event->getEventType()];
                        //print_r($list);
                        if (!empty($list)) {
                                foreach ($this->listenersByEventType[$event->getEventType()] as $k => $observer_id) {
                                        //print_r($list);
                                        $class = get_class( $this->listeners[$observer_id][0] );
                                        $responses[ $class ] = call_user_func_array($this->listeners[$observer_id], array($event));
                                        //owa_coreAPI::debug(print_r($event, true));
                                        owa_coreAPI::debug(sprintf("%s event handled by %s.",$event->getEventType(), get_class($this->listeners[$observer_id][0])));
                                }
                        }
                } else {
                        owa_coreAPI::debug("no listeners registered for this event type.");
                }       
                
                owa_coreAPI::debug('EHS: Responses - '.print_r($responses, true));
                
                if ( in_array( OWA_EHS_EVENT_FAILED, $responses, true ) ) {
                        owa_coreAPI::debug("EHS: Event was not handled successfully by some handlers.");
                        //$q = $this->getAsyncEventQueue(owa_coreAPI::getSetting('base', 'event_queue_type'));
                        //$q->addToQueue($event);
                        return OWA_EHS_EVENT_FAILED;
                } else {
                        owa_coreAPI::debug("EHS: Event was handled successfully by all handlers.");
                        return OWA_EHS_EVENT_HANDLED;
                }
                
        }
        
        /**
         * Notify Untill
         *
         * Notifies all handlers of events in order that they were registered
         * Stops notifying after first handler returns true
         * 
         * @param       $event_type     string
         * @param       $event  array
         * @return bool
         */
 
        function notifyUntill() {
                owa_coreAPI::debug("Notifying Until listener for $event_type answers");
        }
        
        /**
         * Filter
         *
         * Filters event by handlers in order that they were registered
         * 
         * @param       $filter_name    string
         * @param       $value  array
         * @return $new_value   mixed
         */
        function filter($filter_name, $value = '') {
                owa_coreAPI::debug("Filtering $filter_name");
                
                if (array_key_exists($filter_name, $this->listenersByFilterType)) {
                        // sort the filter list by priority
                        ksort($this->listenersByFilterType[$filter_name]);
                        //get the function arguments
                        $args = func_get_args();
                        // outer priority loop
                        foreach ($this->listenersByFilterType[$filter_name] as $priority) {
                                // inner filter class/function loop
                                foreach ($priority as $observer_id) {
                                        // pass args to filter
                                        owa_coreAPI::debug(sprintf("Filter: %s::%s. Value passed: %s", get_class($this->listeners[$observer_id][0]),$this->listeners[$observer_id][1], print_r($value, true)));
                                        $value = call_user_func_array($this->listeners[$observer_id], array_slice($args,1));
                                        owa_coreAPI::debug(sprintf("Filter: %s::%s. Value returned: %s", get_class($this->listeners[$observer_id][0]),$this->listeners[$observer_id][1], print_r($value, true)));
                                        // set filterred value as value in args for next filter
                                        $args[1] = $value;
                                        // debug whats going on
                                        owa_coreAPI::debug(sprintf("%s filtered by %s.", $filter_name, get_class($this->listeners[$observer_id][0])));
                                }
                        }
                }
                
                return $value;
        }
        
        /**
         * Log
         *
         * Notifies handlers of tracking events
         * Provides switch for async notification
         * 
         * @param       $event_params   array
         * @param       $event_type     string
         */
        function log($event_params, $event_type = '') {
                //owa_coreAPI::debug("Notifying listeners of tracking event type: $event_type");
                
                if (!is_a($event_params,'owa_event')) {
                        $event = owa_coreAPI::supportClassFactory('base', 'event');
                        $event->setProperties($event_params);
                        $event->setEventType($event_type);
                } else {
                        $event = $event_params;
                }
                
                $this->asyncNotify($event);
                        
        }
        
        /**
         * Async Notify
         *
         * Adds event to async notiication queue for notification by another process.
         * 
         * @param       $event  array
         * @return bool
         */
        function asyncNotify($event) {
                
                // check config to see if async mode is enabled, if not fall back to realtime notification
                if (owa_coreAPI::getSetting('base', 'queue_events')) {
                        owa_coreAPI::debug(sprintf("Adding event of %s to async %s queue.", $event->getEventType(), owa_coreAPI::getSetting('base', 'event_queue_type')));
                        // check to see first if OWA is not already acting as a remote event queue, 
                        // then check to see if we are configured to use a remote or local event queue
                        // then see if we have an endpoint
                        if (!owa_coreAPI::getSetting('base', 'is_remote_event_queue') && 
                            owa_coreAPI::getSetting('base', 'use_remote_event_queue') &&
                            owa_coreAPI::getSetting('base', 'remote_event_queue_type') &&
                            owa_coreAPI::getSetting('base', 'remote_event_queue_endpoint')) {
                            // get a network queue
                                $q = $this->getAsyncEventQueue(owa_coreAPI::getSetting('base', 'remote_event_queue_type'));
                        // use a local event queue
                        } else {
                                // get a local event queue
                                $q = $this->getAsyncEventQueue(owa_coreAPI::getSetting('base', 'event_queue_type'));
                        }
                        
                        // if an event queue is returned then pass it the event
                        if ($q) {
                        
                                return $q->addToQueue($event);
                        // otherwise skip the queue and just notify the listeners immeadiately. 
                        } else {
                                return $this->notify($event);
                        }
                        
                // otherwise skip the queue and just notify the listeners immeadiately.         
                } else {
                        return $this->notify($event);
                }       
        }
        
        function getAsyncEventQueue($type) {
        
                static $q = array();
                
                if ( ! array_key_exists( $type, $q ) ) {
                        
                        switch( $type ) {
                                
                                case 'http':
                                        $q['http'] = owa_coreAPI::supportClassFactory( 'base', 'httpEventQueue' );
                                        break;
                                case 'database':
                                        $q['database'] = owa_coreAPI::supportClassFactory( 'base', 'dbEventQueue' );
                                        break;
                                case 'file':
                                        $q['file'] = owa_coreAPI::supportClassFactory( 'base', 'fileEventQueue' );
                                        break;
                        }
                }               
                
                if ( array_key_exists( $type, $q ) ) {
                        return $q[$type];
                } else {
                        owa_coreAPI::debug('No event queue of that type exists.'); 
                        return false;
                }
        }
        
        function eventFactory() {
                
                return owa_coreAPI::supportClassFactory('base', 'event');
        }
        
        function makeEvent($type = ''