Proyectos de Subversion LeadersLinked - Backend

Rev

Rev 16769 | Rev 16979 | Ir a la última revisión | | Comparar con el anterior | Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
16766 efrain 1
<?php
2
declare(strict_types=1);
3
 
4
namespace LeadersLinked\Controller;
5
 
6
use Laminas\Db\Adapter\AdapterInterface;
16768 efrain 7
 
16766 efrain 8
use Laminas\Mvc\Controller\AbstractActionController;
9
use Laminas\Log\LoggerInterface;
10
use Laminas\View\Model\ViewModel;
11
use LeadersLinked\Mapper\UserMapper;
12
use LeadersLinked\Mapper\CompanyMapper;
13
use LeadersLinked\Mapper\CompanyUserMapper;
14
use LeadersLinked\Mapper\ConversationMapper;
15
use LeadersLinked\Mapper\MessageMapper;
16
use LeadersLinked\Model\Conversation;
17
use LeadersLinked\Library\Image;
18
use Laminas\View\Model\JsonModel;
19
use LeadersLinked\Form\InMail\SendForm;
20
use LeadersLinked\Model\Message;
21
use LeadersLinked\Model\VideoConvert;
22
use LeadersLinked\Mapper\VideoConvertMapper;
23
use Laminas\I18n\Translator\Translator;
24
use LeadersLinked\Library\Functions;
25
 
26
class InMailCompanyController extends AbstractActionController
27
{
28
    /**
29
     *
16769 efrain 30
     * @var \Laminas\Db\Adapter\AdapterInterface
16766 efrain 31
     */
32
    private $adapter;
16769 efrain 33
 
16766 efrain 34
    /**
35
     *
16769 efrain 36
     * @var \LeadersLinked\Cache\CacheInterface
16766 efrain 37
     */
16769 efrain 38
    private $cache;
39
 
40
 
41
    /**
42
     *
43
     * @var \Laminas\Log\LoggerInterface
44
     */
16766 efrain 45
    private $logger;
16769 efrain 46
 
16766 efrain 47
    /**
48
     *
49
     * @var array
50
     */
51
    private $config;
52
 
16769 efrain 53
 
16766 efrain 54
    /**
16769 efrain 55
     *
56
     * @var \Laminas\Mvc\I18n\Translator
16766 efrain 57
     */
58
    private $translator;
16769 efrain 59
 
60
 
16766 efrain 61
    /**
62
     *
16769 efrain 63
     * @param \Laminas\Db\Adapter\AdapterInterface $adapter
64
     * @param \LeadersLinked\Cache\CacheInterface $cache
65
     * @param \Laminas\Log\LoggerInterface LoggerInterface $logger
16766 efrain 66
     * @param array $config
16769 efrain 67
     * @param \Laminas\Mvc\I18n\Translator $translator
16766 efrain 68
     */
16769 efrain 69
    public function __construct($adapter, $cache, $logger, $config, $translator)
16766 efrain 70
    {
71
        $this->adapter      = $adapter;
16769 efrain 72
        $this->cache        = $cache;
16766 efrain 73
        $this->logger       = $logger;
74
        $this->config       = $config;
16769 efrain 75
        $this->translator   = $translator;
16766 efrain 76
    }
77
 
78
    /**
79
     *
80
     * Generación del listado de conversationes
81
     *
82
     * [
83
     *  success: true,
84
     *  data:[
85
     *    [
86
     *       uuid: uuid con quien se tiene la conversación,
87
     *       name: nombre de con quien se tiene la conversacion,
88
     *       image: imagen de con quien se tiene la conversación,
89
     *       profile: url del profile con quien se tiene la conversación,
90
     *       last_message: fecha del ultimo mensaje,
91
     *       count_unread: cantidad de mensajes sin leer,
92
     *       messages_link: url para recuperar los mensajes,
93
     *       send_link: url para enviar el mensaje,
94
     *       selected: 0 = no seleccionado, 1 = seleccionado
95
     *     ]
96
     *  ]
97
     *
98
     *
99
     * @return \Laminas\View\Model\JsonModel
100
     */
101
    public function indexAction()
102
    {
103
        $request = $this->getRequest();
104
        if ($request->isGet()) {
105
            $currentUserPlugin = $this->plugin('currentUserPlugin');
106
            $currentCompany = $currentUserPlugin->getCompany();
107
 
108
            $companyUserMapper = CompanyUserMapper::getInstance($this->adapter);
109
            $companyUser = $companyUserMapper->fetchOwnerByCompanyId($currentCompany->id);
110
 
111
            $userMapper = UserMapper::getInstance($this->adapter);
112
            $currentUser = $userMapper->fetchOne($companyUser->user_id);
113
 
114
            $headers  = $request->getHeaders();
115
 
116
            $isJson = false;
117
            if ($headers->has('Accept')) {
118
                $accept = $headers->get('Accept');
119
 
120
                $prioritized = $accept->getPrioritized();
121
 
122
                foreach ($prioritized as $key => $value) {
123
                    $raw = trim($value->getRaw());
124
 
125
                    if (!$isJson) {
126
                        $isJson = strpos($raw, 'json');
127
                    }
128
                }
129
            }
130
 
131
            //$isJson = true;
132
            if ($isJson) {
133
                $search         = Functions::sanitizeFilterString($this->params()->fromQuery('search'));
134
                $conversation_id = Functions::sanitizeFilterString($this->params()->fromQuery('conversation_id'));
135
                $last_message   = Functions::sanitizeFilterString($this->params()->fromQuery('last_message'));
136
 
137
 
138
                $now = $userMapper->getDatebaseNow();
139
 
140
 
141
 
142
                $conversationMapper = ConversationMapper::getInstance($this->adapter);
143
                $messageMapper = MessageMapper::getInstance($this->adapter);
144
 
145
                $conversations = [];
146
                $messages = [];
147
 
148
                if($conversation_id ) {
149
 
150
 
151
 
152
 
153
 
154
                    $companyMapper = CompanyMapper::getInstance($this->adapter);
155
                    $company = $companyMapper->fetchOneByUuid($conversation_id);
156
                    if ($company) {
157
 
158
 
159
                        $companyUserMapper = CompanyUserMapper::getInstance($this-> adapter);
160
                        $companyUser = $companyUserMapper->fetchOwnerByCompanyId($company->id);
161
                        $user  = $userMapper->fetchOne($companyUser->user_id);
162
 
163
 
164
                        $conversation_id = $user->id;
165
                        $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
166
 
167
 
168
                        $lastMessage = '';
169
                        $timeElapsedString = '';
170
                        if ($conversation) {
171
                            $unread = $messageMapper->fetchCountUnreadMessagesByConversationIdAndReceiverId($conversation->id, $currentUser->id);
172
                            $lastMessage = $messageMapper->fetchLastMessageByConversation($conversation->id);
173
 
174
                            if ($lastMessage) {
175
                                $timeElapsedString =  $this->timeAgo($lastMessage, $now);
176
                            } else {
177
                                $lastMessage = '';
178
                            }
179
                        } else {
180
                            $unread = 0;
181
                        }
182
 
183
 
184
                        array_push($conversations, [
185
                            'uuid' => $company->uuid,
186
                            'name' => $company->name,
187
                            'image' => $this->url()->fromRoute('storage', ['type' => 'company', 'code' => $company->uuid, 'filename' => $company->image]),
188
                            'profile' => '', // $this->url()->fromRoute('company/view', ['id' => $company->uuid]),
189
                            'last_message' => $timeElapsedString,
190
                            'datetime_last_message' => $lastMessage,
191
                            'count_unread' => $unread,
192
                            'messages_link' => $this->url()->fromRoute('inmail-company/messages', ['id' => $user->uuid]),
193
                            'send_link' => $this->url()->fromRoute('inmail-company/messages/send', ['id' => $user->uuid]),
194
                            'selected' => 1,
195
                            'delete_link' => $this->url()->fromRoute('inmail-company/delete',  ['id' => $user->uuid]),
196
                        ]);
197
                    } else {
198
                        $user = $userMapper->fetchOneByUuid($conversation_id);
199
                        if ($user) {
200
 
201
                            $lastMessage = '';
202
                            $timeElapsedString = '';
203
 
204
                            $conversation_id = $user->id;
205
                            $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
206
                            if ($conversation) {
207
                                $unread = $messageMapper->fetchCountUnreadMessagesByConversationIdAndReceiverId($conversation->id, $currentUser->id);
208
                                $lastMessage = $messageMapper->fetchLastMessageByConversation($conversation->id);
209
                                if ($lastMessage) {
210
                                    $timeElapsedString =  $this->timeAgo($lastMessage, $now);
211
                                } else {
212
 
213
                                    $lastMessage = '';
214
                                }
215
                            } else {
216
                                $unread = 0;
217
                            }
218
 
219
 
220
                            array_push($conversations, [
221
                                'uuid' => $user->uuid,
222
                                'name' => trim($user->first_name . ' ' . $user->last_name),
223
                                'image' => $this->url()->fromRoute('storage', ['type' => 'user', 'code' => $user->uuid, 'filename' => $user->image]),
224
                                'profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
225
                                'last_message' => $timeElapsedString,
226
                                'datetime_last_message' => $lastMessage,
227
                                'count_unread' => $unread,
228
                                'messages_link' => $this->url()->fromRoute('inmail-company/messages', ['id' => $user->uuid]),
229
                                'send_link' => $this->url()->fromRoute('inmail-company/messages/send', ['id' => $user->uuid]),
230
                                'selected' => 1,
231
                                'delete_link' => $this->url()->fromRoute('inmail-company/delete',  ['id' => $user->uuid]),
232
                            ]);
233
                        }
234
                    }
235
 
236
                    if($user) {
237
 
238
 
239
                        $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
240
                        if(!$conversation) {
241
                            $conversation = new Conversation();
242
                            $conversation->sender_id = $currentUser->id;
243
                            $conversation->sender_status = Conversation::STATUS_NORMAL;
244
                            $conversation->receiver_id = $user->id;
245
                            $conversation->receiver_status = Conversation::STATUS_NORMAL;
246
                            $conversationMapper->insert($conversation);
247
                        }
248
 
249
 
250
                        $message = $messageMapper->fetchOneByUuid( $last_message);
251
                        if($message && $message->conversation_id == $conversation_id) {
252
                            $message_id = $message->id;
253
                        } else {
254
                            $message_id = 0;
255
                        }
256
 
257
                        $records = $messageMapper->fetchBatchMessagesByConversationIdAndGreaterThanMessageId($conversation->id, $message_id);
258
                        foreach ($records as $record)
259
                        {
260
                            $timeElapsedString = $this->timeAgo($record->added_on, $now);
261
 
262
 
263
                            if ($record->sender_id == $currentUser->id) {
264
                                array_push($messages, [
265
                                    'uuid' => $record->uuid,
266
                                    'sender_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
267
                                    'sender_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $currentUser->image, 'code' => $currentUser->uuid]),
268
                                    'sender_profile' =>'', // $this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
269
                                    'receiver_name' => trim($user->first_name . ' ' . $user->last_name),
270
                                    'receiver_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $user->image, 'code' => $user->uuid]),
271
                                    'receiver_profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
272
                                    'side' => 'left',
273
                                    'message' => $record->message,
274
                                    'type' => $record->type,
275
                                    'filename' => $record->filename ? $this->url()->fromRoute('storage', ['type' => 'message', 'filename' => $record->filename, 'code' => $record->uuid]) : '',
276
                                    'date' => $timeElapsedString
277
                                ]);
278
                            } else {
279
 
280
 
281
 
282
                                array_push($messages, [
283
                                    'uuid' => $record->uuid,
284
                                    'sender_name' => trim($user->first_name . ' ' . $user->last_name),
285
                                    'sender_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $user->image, 'code' => $user->uuid]),
286
                                    'sender_profile' => '', // $this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
287
                                    'receiver_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
288
                                    'receiver_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $currentUser->image, 'code' => $user->uuid]),
289
                                    'receiver_profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
290
                                    'side' => 'right',
291
                                    'message' => $record->message,
292
                                    'type' => $record->type,
293
                                    'filename' => $record->filename ? $this->url()->fromRoute('storage', ['type' => 'message', 'filename' => $record->filename, 'code' => $record->uuid]) : '',
294
                                    'date' => $timeElapsedString
295
                                ]);
296
 
297
                                $messageMapper->markAsRead($record->id);
298
                            }
299
                        }
300
                    }
301
                }
302
 
303
 
304
                $records = $conversationMapper->fetchAllByUserId($currentUser->id);
305
                foreach ($records as $record)
306
                {
307
                    if ($conversation_id) {
308
                        if ($record->sender_id == $currentUser->id && $record->receiver_id == $conversation_id) {
309
                            continue;
310
                        }
311
                        if ($record->receiver_id == $currentUser->id && $record->sender_id == $conversation_id) {
312
                            continue;
313
                        }
314
                    }
315
 
316
                    if ($record->sender_id == $currentUser->id) {
317
                        $user = $userMapper->fetchOne($record->receiver_id);
318
 
319
 
320
 
321
                    }
322
                    if ($record->receiver_id == $currentUser->id) {
323
                        $user = $userMapper->fetchOne($record->sender_id);
324
                    }
325
 
326
 
327
                    $username = trim($user->first_name . ' ' . $user->last_name);
328
 
329
                    if($search) {
330
 
331
 
332
                        if(strpos($username, $search) === false && strpos($user->email, $search) === false) {
333
 
334
                            continue;
335
                        }
336
 
337
                    }
338
 
339
                    $timeElapsedString = '';
340
 
341
 
342
                    $unread = $messageMapper->fetchCountUnreadMessagesByConversationIdAndReceiverId($record->id, $currentUser->id);
343
                    $lastMessage = $messageMapper->fetchLastMessageByConversation($record->id);
344
 
345
 
346
                    if ($lastMessage) {
347
                        $timeElapsedString =  $this->timeAgo($lastMessage, $now);
348
                    } else {
349
                        $lastMessage = '';
350
                    }
351
 
352
                    array_push($conversations, [
353
                        'uuid' => $user->uuid,
354
                        'name' => $username,
355
                        'image' => $this->url()->fromRoute('storage', ['type' => 'user', 'code' => $user->uuid, 'filename' => $user->image]),
356
                        'profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
357
                        'last_message' => $timeElapsedString,
358
                        'datetime_last_message' => $lastMessage,
359
                        'count_unread' => $unread,
360
                        'messages_link' => $this->url()->fromRoute('inmail-company/messages', ['id' => $user->uuid]),
361
                        'send_link' => $this->url()->fromRoute('inmail-company/messages/send', ['id' => $user->uuid]),
362
                        'selected' => $conversation_id == $user->id ? 1 : 0,
363
                        'delete_link' => $this->url()->fromRoute('inmail-company/delete',  ['id' => $user->uuid]),
364
                    ]);
365
                }
366
 
367
 
368
                usort($conversations, function ($a, $b) {
369
                    if ($a['selected'] == $b['selected']) {
370
                       return $a['datetime_last_message'] < $b['datetime_last_message'] ? 1 : -1;
371
                    } else {
372
                        return $a['selected'] < $b['selected'] ? 1 : 0;
373
                    }
374
                });
375
 
376
 
377
                return new JsonModel([
378
                    'success' => true,
379
                    'data' => [
380
                        'conversations' => $conversations,
381
                        'messages' => $messages,
382
                    ],
383
                ]);
384
            } else {
385
                $this->layout()->setTemplate('layout/layout-backend');
386
                $viewModel = new ViewModel();
387
                $viewModel->setTemplate('leaders-linked/inmail/company.phtml');
388
                return $viewModel;
389
            }
390
        } else {
391
            return new JsonModel([
392
                'success' => false,
393
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
394
            ]);
395
        }
396
    }
397
 
398
    public function startConversationAction()
399
    {
400
        $request = $this->getRequest();
401
        if ($request->isPost()) {
402
            $currentUserPlugin = $this->plugin('currentUserPlugin');
403
            $currentUser = $currentUserPlugin->getUser();
404
 
405
            $userMapper = UserMapper::getInstance($this->adapter);
406
            $now = $userMapper->getDatebaseNow();
407
 
408
 
409
            $id = Functions::sanitizeFilterString($this->params()->fromPost('id'));
410
            if (!$id) {
411
                return new JsonModel([
412
                    'success' => false,
413
                    'data' => 'ERROR_PARAMETERS_ARE_INVALID'
414
                ]);
415
            }
416
 
417
 
418
            $user = $userMapper->fetchOneByUuidAndNetworkId($id, $currentUser->network_id);
419
            if (!$user) {
420
                return new JsonModel([
421
                    'success' => false,
422
                    'data' => 'ERROR_REQUEST_IS_INVALID'
423
                ]);
424
            }
425
 
426
 
427
            $conversationMapper = ConversationMapper::getInstance($this->adapter);
428
            $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
429
            if(!$conversation) {
430
                $conversation = new Conversation();
431
                $conversation->sender_id = $currentUser->id;
432
                $conversation->sender_status = Conversation::STATUS_NORMAL;
433
                $conversation->receiver_id = $user->id;
434
                $conversation->receiver_status = Conversation::STATUS_NORMAL;
435
                $conversationMapper->insert($conversation);
436
            } else {
437
                if($conversation->sender_id == $currentUser->id) {
438
                    $conversation->sender_status = Conversation::STATUS_NORMAL;
439
                } else {
440
                    $conversation->sender_status = Conversation::STATUS_NORMAL;
441
                }
442
                $conversationMapper->update($conversation);
443
            }
444
 
445
            $timeElapsedString = '';
446
 
447
 
448
            $messageMapper = MessageMapper::getInstance($this->adapter);
449
            $unread = $messageMapper->fetchCountUnreadMessagesByConversationIdAndReceiverId($conversation->id, $currentUser->id);
450
            $lastMessage = $messageMapper->fetchLastMessageByConversation($conversation->id);
451
 
452
 
453
            if ($lastMessage) {
454
                $timeElapsedString =  $this->timeAgo($lastMessage, $now);
455
            } else {
456
                $lastMessage = '';
457
            }
458
 
459
            return new JsonModel([
460
                'success' => true,
461
                'data' => [
462
                    'uuid' => $user->uuid,
463
                    'name' => trim($user->first_name . ' ' . $user->last_name),
464
                    'image' => $this->url()->fromRoute('storage', ['type' => 'user', 'code' => $user->uuid, 'filename' => $user->image]),
465
                    'profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
466
                    'last_message' => $timeElapsedString,
467
                    'datetime_last_message' => $lastMessage,
468
                    'count_unread' => $unread,
469
                    'messages_link' => $this->url()->fromRoute('inmail-personal/messages', ['id' => $user->uuid]),
470
                    'send_link' => $this->url()->fromRoute('inmail-personal/messages/send', ['id' => $user->uuid]),
471
                    'selected' => 1,
472
                    'delete_link' => $this->url()->fromRoute('inmail-personal/delete',  ['id' => $user->uuid]),
473
                ]
474
 
475
            ]);
476
 
477
 
478
 
479
 
480
        } else {
481
            return new JsonModel([
482
                'success' => false,
483
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
484
            ]);
485
        }
486
    }
487
 
488
 
489
    /**
490
     * Esta función remueve el usuario del grupo
491
     * Es una petición GET el url recuperado en /inmail
492
     * [ruta del servidor]/inmail-company/[uuid]/message
493
     * retorna un json en caso de ser  positivo
494
     * [
495
     *   success: true,
496
     *      data: [
497
     *          [
498
     *              sender_name: nombre de quien que envia el mensaje,
499
     *              sender_image: imagen de quien que recibe el mensaje,
500
     *              sender_profile: url de profile de quien que envia el mensaje,
501
     *              receiver_name: nombre de quien que recibe el mensaje,
502
     *              receiver_image: url de la imagen de quien  que recibe el mensaje,
503
     *              receiver_profile: url del profile de quien recibe el mensaje,
504
     *              side: left/righ de que lado de la pantalla se mostrara,
505
     *              message: texto del mensaje,
506
     *              type: text|image|video|document,
507
     *              filename: url del archivo enviado,
508
     *              date: cadena que describe hace cuanto se recibio el mensaje
509
     *          ]
510
     *      ]
511
     *   ]
512
     * En caso de ser negativo puede haber 2 formatos
513
     * [
514
     *  'success' : false,
515
     *  'data' : mensaje de error
516
     * ]
517
     * @return \Laminas\View\Model\JsonModel
518
     */
519
    public function messageAction()
520
    {
521
 
522
 
523
        $request = $this->getRequest();
524
        if ($request->isGet()) {
525
            $currentUserPlugin = $this->plugin('currentUserPlugin');
526
            $currentCompany = $currentUserPlugin->getCompany();
527
 
528
            $companyUserMapper = CompanyUserMapper::getInstance($this->adapter);
529
            $companyUser = $companyUserMapper->fetchOwnerByCompanyId($currentCompany->id);
530
 
531
            $userMapper = UserMapper::getInstance($this->adapter);
532
            $currentUser = $userMapper->fetchOne($companyUser->user_id);
533
 
534
 
535
            $now = $userMapper->getDatebaseNow();
536
 
537
            $conversationMapper = ConversationMapper::getInstance($this->adapter);
538
 
539
            $messageMapper = MessageMapper::getInstance($this->adapter);
540
 
541
 
542
            $type   = Functions::sanitizeFilterString($this->params()->fromQuery('type', ''));
543
            if(!in_array($type, ['first','changes', 'older'])) {
544
                $type = 'first';
545
            }
546
 
547
            $id = $this->params()->fromRoute('id');
548
            if (!$id) {
549
                return new JsonModel([
550
                    'success' => false,
551
                    'data' => 'ERROR_PARAMETERS_ARE_INVALID'
552
                ]);
553
            }
554
 
555
            $user = $userMapper->fetchOneByUuid($id);
556
            if (!$user) {
557
                return new JsonModel([
558
                    'success' => false,
559
                    'data' => 'ERROR_REQUEST_IS_INVALID'
560
                ]);
561
            }
562
 
563
 
564
 
565
            $first_message = Functions::sanitizeFilterString($this->params()->fromQuery('first_message'));
566
            $last_message = Functions::sanitizeFilterString($this->params()->fromQuery('last_message'));
567
 
568
 
569
 
570
            $conversationMapper = ConversationMapper::getInstance($this->adapter);
571
            $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
572
            if(!$conversation) {
573
                $conversation = new Conversation();
574
                $conversation->sender_id = $currentUser->id;
575
                $conversation->sender_status = Conversation::STATUS_NORMAL;
576
                $conversation->receiver_id = $user->id;
577
                $conversation->receiver_status = Conversation::STATUS_NORMAL;
578
                $conversationMapper->insert($conversation);
579
            }
580
 
581
 
582
            $messages = [];
583
            if ($conversation) {
584
 
585
                switch($type)
586
                {
587
                    case 'older' :
588
                        $message = $messageMapper->fetchOneByUuid( $first_message);
589
                        if($message && $message->conversation_id == $conversation->id) {
590
                            $message_id = $message->id;
591
                        } else {
592
                            $message_id = 0;
593
                        }
594
                        $records = $messageMapper->fetchBatchMessagesByConversationIdAndLessThanMessageId($conversation->id, $message_id);
595
                        $count  = $messageMapper->fetchCountMessagesByConversationIdAndLessThanMessageId($conversation->id, $message_id) - count( $records );
596
                        break;
597
 
598
                    case 'changes' :
599
                        $message = $messageMapper->fetchOneByUuid( $last_message);
600
                        if($message && $message->conversation_id == $conversation->id) {
601
                            $message_id = $message->id;
602
                        } else {
603
                            $message_id = 0;
604
                        }
605
                        $records = $messageMapper->fetchBatchMessagesByConversationIdAndGreaterThanMessageId($conversation->id, $message_id);
606
                        $count  = $messageMapper->fetchCountMessagesByConversationIdAndGreaterThanMessageId($conversation->id, $message_id) - count( $records );
607
                        break;
608
 
609
                    default :
610
 
611
                        $records = $messageMapper->fetchBatchMessagesByConversationId($conversation->id);
612
                        $count  = $messageMapper->fetchCountMessagesByConversationId($conversation->id) - count( $records );
613
 
614
                        break;
615
 
616
                }
617
 
618
                foreach ($records as $record)
619
                {
620
                    $timeElapsedString = $this->timeAgo($record->added_on, $now);
621
 
622
 
623
                    if ($record->sender_id == $currentUser->id) {
624
                        array_push($messages, [
625
                            'uuid' => $record->uuid,
626
                            'sender_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
627
                            'sender_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $currentUser->image, 'code' => $currentUser->uuid]),
628
                            'sender_profile' =>'', // $this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
629
                            'receiver_name' => trim($user->first_name . ' ' . $user->last_name),
630
                            'receiver_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $user->image, 'code' => $user->uuid]),
631
                            'receiver_profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
632
                            'side' => 'left',
633
                            'message' => $record->message,
634
                            'type' => $record->type,
635
                            'filename' => $record->filename ? $this->url()->fromRoute('storage', ['type' => 'message', 'filename' => $record->filename, 'code' => $record->uuid]) : '',
636
                            'date' => $timeElapsedString
637
                        ]);
638
                    } else {
639
 
640
 
641
 
642
                        array_push($messages, [
643
                            'uuid' => $record->uuid,
644
                            'sender_name' => trim($user->first_name . ' ' . $user->last_name),
645
                            'sender_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $user->image, 'code' => $user->uuid]),
646
                            'sender_profile' => '', // $this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
647
                            'receiver_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
648
                            'receiver_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $currentUser->image, 'code' => $user->uuid]),
649
                            'receiver_profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
650
                            'side' => 'right',
651
                            'message' => $record->message,
652
                            'type' => $record->type,
653
                            'filename' => $record->filename ? $this->url()->fromRoute('storage', ['type' => 'message', 'filename' => $record->filename, 'code' => $record->uuid]) : '',
654
                            'date' => $timeElapsedString
655
                        ]);
656
 
657
                        $messageMapper->markAsRead($record->id);
658
                    }
659
                }
660
            }
661
 
662
 
663
            $timeElapsedString = '';
664
            $unread = $messageMapper->fetchCountUnreadMessagesByConversationIdAndReceiverId($user->id, $currentUser->id);
665
            $lastMessage = $messageMapper->fetchLastMessageByConversation($user->id);
666
 
667
 
668
            if ($lastMessage) {
669
                $timeElapsedString =  $this->timeAgo($lastMessage, $now);
670
            }
671
 
672
            return new JsonModel([
673
                'success' => true,
674
                'data' => [
675
                    'messages' => $messages,
676
                    'there_olders' => $count > 0,
677
                    'last_message' => $timeElapsedString,
678
                    'datetime_last_message' => $lastMessage,
679
                    'count_unread' => $unread,
680
                ]
681
            ]);
682
        } else {
683
            return new JsonModel([
684
                'success' => false,
685
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
686
            ]);
687
        }
688
    }
689
 
690
    /**
691
     * Esta función envia un mensaje
692
     * Es una petición POST el url recuperado en /inmail
693
     * [ruta del servidor]/inmail-company/[uidd del usuario]/message/send
694
     * Lo párametros son
695
     * message: text plain requerido
696
     * file: no requerido puede jpg, png, jpeg, mp3, mp4, webm o pdf
697
     * o
698
     * [ruta del servidor]/inmail-company/[uidd del usuario]/message/send/encoding/base64 para el caso de react-native
699
     * Los párametros son
700
     * message: text plain requerido
701
     * fileBase64Name: nombre del archivo  que se envia en base64 solo requerido si se envia un archivo
702
     * fileBase64Content: contenido en base 64 del archivo
703
     *
704
     * retorna un json en caso de ser  positivo
705
     * [
706
     *   success: true,
707
     *      data:{
708
     *          sender_name: nombre de quien que envia el mensaje,
709
     *          sender_image: imagen de quien que recibe el mensaje,
710
     *          sender_profile: url de profile de quien que envia el mensaje,
711
     *          receiver_name: nombre de quien que recibe el mensaje,
712
     *          receiver_image: url de la imagen de quien  que recibe el mensaje,
713
     *          receiver_profile: url del profile de quien recibe el mensaje,
714
     *          side: left/righ de que lado de la pantalla se mostrara,
715
     *          message: texto del mensaje,
716
     *          type: text|image|video|document,
717
     *          filename: url del archivo enviado,
718
     *          date: cadena que describe hace cuanto se recibio el mensaje
719
     *      ]
720
     *   ]
721
     * En caso de ser negativo puede haber 2 formatos
722
     * [
723
     *  'success' : false,
724
     *  'data' : mensaje de error
725
     * ]
726
     * @return \Laminas\View\Model\JsonModel
727
     */
728
    public function sendMessageAction()
729
    {
730
        $request = $this->getRequest();
731
        if ($request->isPost()) {
732
            $currentUserPlugin = $this->plugin('currentUserPlugin');
733
            $currentCompany = $currentUserPlugin->getCompany();
734
 
735
            $companyUserMapper = CompanyUserMapper::getInstance($this->adapter);
736
            $companyUser = $companyUserMapper->fetchOwnerByCompanyId($currentCompany->id);
737
 
738
            $userMapper = UserMapper::getInstance($this->adapter);
739
            $currentUser = $userMapper->fetchOne($companyUser->user_id);
740
 
741
            $now = $userMapper->getDatebaseNow();
742
 
743
 
744
            $id = $this->params()->fromRoute('id');
745
            if (!$id) {
746
                return new JsonModel([
747
                    'success' => false,
748
                    'data' => 'ERROR_PARAMETERS_ARE_INVALID'
749
                ]);
750
            }
751
 
752
 
753
            $user = $userMapper->fetchOneByUuidAndNetworkId($id, $currentUser->network_id);
754
            if (!$user) {
755
                return new JsonModel([
756
                    'success' => false,
757
                    'data' => 'ERROR_REQUEST_IS_INVALID'
758
                ]);
759
            }
760
 
761
 
762
            $data = array_merge($request->getPost()->toArray(), $request->getFiles()->toArray());
763
 
764
            $form = new SendForm();
765
            $form->setData($data);
766
 
767
            if ($form->isValid()) {
768
 
769
                $dataPost = (array) $form->getData();
770
 
771
                $conversationMapper = ConversationMapper::getInstance($this->adapter);
772
                $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
773
 
774
                if ($conversation) {
775
                    $conversation->receiver_status = Conversation::STATUS_NORMAL;
776
                    $conversation->sender_status = Conversation::STATUS_NORMAL;
777
 
778
                    if (!$conversationMapper->update($conversation)) {
779
                        return new JsonModel([
780
                            'success'  => false,
781
                            'data' => $conversationMapper->getError()
782
                        ]);
783
                    }
784
                } else {
785
                    $conversation = new Conversation();
786
                    $conversation->sender_id = $currentUser->id;
787
                    $conversation->sender_status = Conversation::STATUS_NORMAL;
788
                    $conversation->receiver_id = $user->id;
789
                    $conversation->receiver_status = Conversation::STATUS_NORMAL;
790
 
791
                    if (!$conversationMapper->insert($conversation)) {
792
                        return new JsonModel([
793
                            'success'  => false,
794
                            'data' => $conversationMapper->getError()
795
                        ]);
796
                    }
797
                }
798
                $files = $this->getRequest()->getFiles()->toArray();
799
                $type = Message::TYPE_TEXT;
800
 
801
 
802
                if (isset($files['filename']) && empty($files['filename']['error'])) {
803
                    $message_tmp_filename  = $files['filename']['tmp_name'];
804
                    $message_filename      = $this->normalizeString($files['filename']['name']);
805
 
806
                    $mime_type = mime_content_type($message_tmp_filename);
807
 
808
 
809
                    if ($mime_type == 'image/jpg' || $mime_type == 'image/jpeg' || $mime_type == 'image/png') {
810
                        $type = Message::TYPE_IMAGE;
16822 efrain 811
                    } else if ($mime_type == 'video/quicktime' ||  $mime_type == 'video/webm' || $mime_type == 'video/mpeg' || $mime_type == 'video/mpg' || $mime_type == 'video/mp4') {
16766 efrain 812
                        $type = Message::TYPE_VIDEO;
813
                    } else if ($mime_type == 'application/pdf') {
814
                        $type = Message::TYPE_DOCUMENT;
815
                    }
816
                }
817
 
818
 
819
 
820
 
821
                $message = new Message();
822
                $message->conversation_id = $conversation->id;
823
                $message->read = Message::NO;
824
                $message->message = $type == Message::TYPE_TEXT ? $dataPost['message'] : '';
825
                $message->receiver_id = $user->id;
826
                $message->receiver_status = Message::STATUS_NORMAL;
827
                $message->sender_id = $currentUser->id;
828
                $message->sender_status = Message::STATUS_NORMAL;
829
                $message->type = $type;
830
 
831
 
832
 
833
 
834
 
835
                $messageMapper = MessageMapper::getInstance($this->adapter);
836
                if ($messageMapper->insert($message)) {
837
 
838
                    $message = $messageMapper->fetchOne($message->id);
839
 
840
 
841
                    if ($type == Message::TYPE_DOCUMENT) {
842
                        try {
843
                            $target_path = $this->config['leaderslinked.fullpath.message'] . DIRECTORY_SEPARATOR . $message->uuid;
844
                            if (!file_exists($target_path)) {
845
                                mkdir($target_path, 0755);
846
                            }
847
 
848
                            $full_filename = $target_path  . DIRECTORY_SEPARATOR . $message_filename;
849
                            /*
850
                            if($encoding == 'base64') {
851
                                $resultMoveOrRename = rename($message_tmp_filename , $full_filename);
852
                            } else {
853
                                $resultMoveOrRename = move_uploaded_file($message_tmp_filename , $full_filename);
854
                            }
855
                            */
856
 
857
                            $resultMoveOrRename = move_uploaded_file($message_tmp_filename, $full_filename);
858
                            if ($resultMoveOrRename) {
859
 
860
                                $message->type = $type;
861
                                $message->filename = basename($message_filename);
862
                                $messageMapper->update($message);
863
                            } else {
864
                                error_log('no se pudo mover o renombrar el documento : ' . $message_tmp_filename . ' al directorio : ' . $full_filename);
865
                            }
866
                        } catch (\Throwable $e) {
867
                            error_log($e->getTraceAsString());
868
                        }
869
                    }
870
                    if ($type == Message::TYPE_IMAGE) {
871
                        try {
872
                            $target_path = $this->config['leaderslinked.fullpath.message'] . DIRECTORY_SEPARATOR . $message->uuid;
873
                            if (!file_exists($target_path)) {
874
                                mkdir($target_path, 0755);
875
                            }
876
 
877
                            list($target_width, $target_height) = explode('x', $this->config['leaderslinked.image_sizes.message_image_size']);
878
 
879
                            $message_filename = substr($message_filename, 0, strrpos($message_filename, '.'))  . '.png';
880
 
881
                            //echo "target_path = $target_path message_tmp_filename = $message_tmp_filename message_filename = $message_filename"; exit;
882
 
883
                            $crop_to_dimensions = false;
884
                            if (Image::uploadImage($message_tmp_filename, $target_path, $message_filename, $target_width, $target_height, $crop_to_dimensions)) {
885
                                $message->type = $type;
886
                                $message->filename = basename($message_filename);
887
                                $messageMapper->update($message);
888
                            }
889
                        } catch (\Throwable $e) {
890
                            error_log($e->getTraceAsString());
891
                        }
892
                    }
893
                    if ($type == Message::TYPE_VIDEO) {
894
                        try {
895
                            $target_path = $this->config['leaderslinked.fullpath.message'] . DIRECTORY_SEPARATOR . $message->uuid;
896
                            if (!file_exists($target_path)) {
897
                                mkdir($target_path, 0755);
898
                            }
899
 
900
                            $full_filename = $target_path  . DIRECTORY_SEPARATOR . $message_filename;
901
 
902
                            /*
903
                            if($encoding == 'base64') {
904
                                $resultMoveOrRename = rename($message_tmp_filename , $full_filename);
905
                            } else {
906
                                $resultMoveOrRename = move_uploaded_file($message_tmp_filename , $full_filename);
907
                            }*/
908
 
909
                            $resultMoveOrRename = move_uploaded_file($message_tmp_filename, $full_filename);
910
                            if ($resultMoveOrRename) {
911
 
912
                                $size = $this->config['leaderslinked.image_sizes.message_image_size'];
913
                                $getFromSecound = 2;
914
 
915
                                //extracción del cover
916
                                $generateFileName = substr($message_filename, 0, strrpos($message_filename, '.'));
917
                                $generateFile =  $target_path  . DIRECTORY_SEPARATOR . $generateFileName .  '.png';
918
                                $cmd            = "/usr/bin/ffmpeg -i $full_filename -an -ss $getFromSecound -s $size $generateFile";
919
                                exec($cmd);
920
 
921
 
922
                                $message->type = $type;
923
                                $message->filename = basename($message_filename);
924
                                $message->image_preview = basename($generateFile);
925
                                $messageMapper->update($message);
926
 
927
                                $videoConvert = new VideoConvert();
928
                                $videoConvert->filename = $full_filename;
929
                                $videoConvert->type = VideoConvert::TYPE_FEED;
930
 
931
                                $videoConvertMapper = VideoConvertMapper::getInstance($this->adapter);
932
                                $videoConvertMapper->insert($videoConvert);
933
                            } else {
934
                                error_log('no se pudo mover o renombrar el documento : ' . $message_tmp_filename . ' al directorio : ' . $full_filename);
935
                            }
936
                        } catch (\Throwable $e) {
937
                            error_log($e->getTraceAsString());
938
                        }
939
                    }
940
 
941
                    $message = $messageMapper->fetchOne($message->id);
942
 
943
                    if ($message->filename) {
944
                        $filename = $this->url()->fromRoute('storage', ['type' => 'message', 'filename' => $message->filename, 'code' => $message->uuid]);
945
                    } else {
946
                        $filename = '';
947
                    }
948
 
949
                    //$dt = \DateTime::createFromFormat('Y-m-d H:i:s', $message->added_on);
950
 
951
                    $userMapper->updateLastActivity($currentUser->id);
952
 
953
 
954
                    $count = $messageMapper->fetchCountMessagesByConversationId($conversation->id);
955
 
956
                    if($count < 10) {
957
                        $max_page = 1;
958
                    } else {
959
                        $max_page = (int) ($count / 10);
960
                        if($count > ($max_page * 10) ) {
961
                            $max_page++;
962
                        }
963
                    }
964
 
965
 
966
                    return new JsonModel([
967
                        'success'   => true,
968
                        'data'   => [
969
                            'max_page' => $max_page,
970
                            'message' => [
971
                                'uuid' => $message->uuid,
972
                                'sender_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
973
                                'sender_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $currentUser->image, 'code' => $currentUser->uuid]),
974
                                'sender_profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
975
                                'receiver_name' => trim($user->first_name . ' ' . $user->last_name),
976
                                'receiver_image' => $this->url()->fromRoute('storage', ['type' => 'user', 'filename' => $user->image, 'code' => $user->uuid]),
977
                                'receiver_profile' => '', //$this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
978
                                'side' => 'left',
979
                                'message' => $message->message,
980
                                'type' => $message->type,
981
                                'filename' => $filename,
982
                                'date' => $this->timeAgo($now, $now),
983
                            ]
984
                        ]
985
                    ]);
986
                } else {
987
                    return new JsonModel([
988
                        'success'   => false,
989
                        'data'   => $messageMapper->getError()
990
                    ]);
991
                }
992
            } else {
993
 
994
                $message = '';
995
                $form_messages = (array) $form->getMessages();
996
                foreach ($form_messages  as $fieldname => $field_messages) {
997
                    $messages = array_values($field_messages);
998
                    $message = array_pop($messages);
999
                }
1000
 
1001
                return new JsonModel([
1002
                    'success'   => false,
1003
                    'data'   => $message,
1004
                ]);
1005
            }
1006
        } else {
1007
            return new JsonModel([
1008
                'success' => false,
1009
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
1010
            ]);
1011
        }
1012
    }
1013
 
1014
 
1015
 
1016
    private function normalizeString($str = '')
1017
    {
1018
        $basename  = substr($str, 0, strrpos($str, '.'));
1019
        $basename  = str_replace('.', '-', $basename);
1020
 
1021
        $extension  = substr($str, strrpos($str, '.'));
1022
 
1023
        $str = $basename . $extension;
1024
 
1025
        $str = strip_tags($str);
1026
        $str = preg_replace('/[\r\n\t ]+/', ' ', $str);
1027
        $str = preg_replace('/[\"\*\/\:\<\>\?\'\|\,]+/', ' ', $str);
1028
        $str = strtolower($str);
1029
        $str = html_entity_decode($str, ENT_QUOTES, "utf-8");
1030
        $str = htmlentities($str, ENT_QUOTES, "utf-8");
1031
        $str = preg_replace("/(&)([a-z])([a-z]+;)/i", '$2', $str);
1032
        $str = str_replace(' ', '-', $str);
1033
        $str = rawurlencode($str);
1034
        $str = str_replace('%', '-', $str);
1035
        return trim(strtolower($str));
1036
    }
1037
 
1038
    /**
1039
     *
1040
     * @param string $timestamp
1041
     * @param string $now
1042
     * @return string
1043
     */
1044
    private function timeAgo($timestamp, $now = '')
1045
    {
1046
 
1047
        if ($now) {
1048
            $datetime1 = \DateTime::createFromFormat('Y-m-d H:i:s', $now);
1049
        } else {
1050
            $now = date('Y-m-d H:i:s');
1051
            $datetime1 = date_create($now);
1052
        }
1053
        $datetime2 = date_create($timestamp);
1054
 
1055
        $diff = date_diff($datetime1, $datetime2);
1056
        $timemsg = '';
1057
        if ($diff->y > 0) {
1058
            $timemsg = $diff->y . ' ' .($diff->y > 1 ? $this->translator->translate('LABEL_YEARS_SMALL')  : $this->translator->translate('LABEL_YEAR_SMALL') );
1059
        } else if ($diff->m > 0) {
1060
            $timemsg = $diff->m  . ' ' .($diff->m > 1 ? $this->translator->translate('LABEL_MONTHS_SMALL')  : $this->translator->translate('LABEL_MONTH_SMALL') );
1061
        } else if ($diff->d > 0) {
1062
            $timemsg = $diff->d . ' ' .($diff->d > 1 ? $this->translator->translate('LABEL_DAYS_SMALL')  : $this->translator->translate('LABEL_DAY_SMALL') );
1063
        } else if ($diff->h > 0) {
1064
            $timemsg = $diff->h  . ' ' .($diff->h > 1 ? $this->translator->translate('LABEL_HOURS_SMALL')  : $this->translator->translate('LABEL_HOUR_SMALL') );
1065
        } else if ($diff->i > 0) {
1066
            $timemsg = $diff->i  . ' ' .($diff->i > 1 ? $this->translator->translate('LABEL_MINUTES_SMALL')  : $this->translator->translate('LABEL_MINUTE_SMALL') );
1067
        } else if ($diff->s > 0) {
1068
            $timemsg = $diff->s  . ' ' . ($diff->s > 1 ? $this->translator->translate('LABEL_SECONDS_SMALL')  : $this->translator->translate('LABEL_SECOND_SMALL') );
1069
        }
1070
        if (!$timemsg) {
1071
            $timemsg = $this->translator->translate('LABEL_NOW') ;
1072
        } else {
1073
            $timemsg = $this->translator->translate('LABEL_AGO_SMALL') . ' '. $timemsg . '';
1074
        }
1075
        return $timemsg;
1076
 
1077
    }
1078
 
1079
 
1080
    public function deleteAction()
1081
    {
1082
        $request = $this->getRequest();
1083
        if ($request->isPost()) {
1084
            $currentUserPlugin = $this->plugin('currentUserPlugin');
1085
            $currentCompany = $currentUserPlugin->getCompany();
1086
 
1087
            $companyUserMapper = CompanyUserMapper::getInstance($this->adapter);
1088
            $companyUser = $companyUserMapper->fetchOwnerByCompanyId($currentCompany->id);
1089
 
1090
            $userMapper = UserMapper::getInstance($this->adapter);
1091
            $currentUser = $userMapper->fetchOne($companyUser->user_id);
1092
 
1093
            $id = $this->params()->fromRoute('id');
1094
            if (!$id) {
1095
                return new JsonModel([
1096
                    'success' => false,
1097
                    'data' => 'ERROR_PARAMETERS_ARE_INVALID'
1098
                ]);
1099
            }
1100
 
1101
            $userMapper = UserMapper::getInstance($this->adapter);
1102
            $user = $userMapper->fetchOneByUuid($id);
1103
            if (!$user) {
1104
                return new JsonModel([
1105
                    'success' => false,
1106
                    'data' => 'ERROR_REQUEST_IS_INVALID'
1107
                ]);
1108
            }
1109
 
1110
            $conversationMapper = ConversationMapper::getInstance($this->adapter);
1111
            $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
1112
 
1113
 
1114
            if ($conversation) {
1115
 
1116
                if ($conversation->sender_id == $currentUser->id && $conversation->receiver_id == $user->id) {
1117
                    $conversation->sender_status = Conversation::STATUS_DELETED;
1118
                    if ($conversationMapper->update($conversation)) {
1119
                        $response = [
1120
                            'success' => true,
1121
                            'data' => 'LABEL_CONVERSATION_WAS_DELETED'
1122
                        ];
1123
                    } else {
1124
 
1125
 
1126
                        $response = [
1127
                            'success' => false,
1128
                            'data' => $conversationMapper->getError()
1129
                        ];
1130
                    }
1131
                }
1132
 
1133
                if ($conversation->receiver_id == $currentUser->id && $conversation->sender_id == $user->id) {
1134
                    $conversation->receiver_status = Conversation::STATUS_DELETED;
1135
                    if ($conversationMapper->update($conversation)) {
1136
                        $response = [
1137
                            'success' => true,
1138
                            'data' => 'LABEL_CONVERSATION_WAS_DELETED'
1139
                        ];
1140
                    } else {
1141
 
1142
 
1143
                        $response = [
1144
                            'success' => false,
1145
                            'data' => $conversationMapper->getError()
1146
                        ];
1147
                    }
1148
                }
1149
 
1150
                return new JsonModel($response);
1151
 
1152
            } else {
1153
                $response = [
1154
                    'success' => false,
1155
                    'data' => 'ERROR_CONVERSATION_NOT_FOUND'
1156
                ];
1157
            }
1158
 
1159
            return new JsonModel($response);
1160
        } else {
1161
            $response = [
1162
                'success' => false,
1163
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
1164
            ];
1165
        }
1166
 
1167
        return new JsonModel($response);
1168
    }
1169
 
1170
 
1171
 
1172
 
1173
}