6 Commits

Author SHA1 Message Date
de81f16557 Logging 2022-12-01 14:15:54 -03:00
c0ddd00cc6 Scheduling UI corrected 2022-11-30 20:45:18 -03:00
5e4e52e620 No registered mailboxes 2022-11-30 11:46:57 -03:00
f6ac8eae7c Fix: Empty mailbox 2022-11-30 11:46:41 -03:00
e42c9f4d8f Fix: Empty mailbox 2022-11-30 11:32:00 -03:00
a5d97729dc Changed way to connect to api 2022-11-30 10:40:36 -03:00
34 changed files with 466 additions and 274 deletions

View File

@ -12,7 +12,7 @@ class Jobs
{ {
use Json; use Json;
public function schedule(ServerRequestInterface $request, ResponseInterface $response, Service $jobsService): ResponseInterface public function schedule(ServerRequestInterface $request, ResponseInterface $response, Service $service, \ProVM\Common\Service\Messages $messagesService): ResponseInterface
{ {
$body = $request->getBody(); $body = $request->getBody();
$json = \Safe\json_decode($body->getContents()); $json = \Safe\json_decode($body->getContents());
@ -25,17 +25,20 @@ class Jobs
'scheduled' => 0 'scheduled' => 0
]; ];
foreach ($json->messages as $message_id) { foreach ($json->messages as $message_id) {
if ($jobsService->schedule($message_id)) { if ($service->schedule($message_id)) {
$message = $messagesService->getRepository()->fetchById($message_id);
$message->doesHaveScheduledDownloads();
$messagesService->getRepository()->save($message);
$output['scheduled'] ++; $output['scheduled'] ++;
} }
} }
return $this->withJson($response, $output); return $this->withJson($response, $output);
} }
public function pending(ServerRequestInterface $request, ResponseInterface $response, Service $jobsService): ResponseInterface public function pending(ServerRequestInterface $request, ResponseInterface $response, Service $service): ResponseInterface
{ {
$pending = array_map(function(Job $job) { $pending = array_map(function(Job $job) {
return $job->toArray(); return $job->toArray();
}, $jobsService->getPending()); }, $service->getPending());
$output = [ $output = [
'total' => count($pending), 'total' => count($pending),
'pending' => $pending 'pending' => $pending

View File

@ -83,16 +83,25 @@ class Mailboxes
$body = $request->getBody(); $body = $request->getBody();
$json = \Safe\json_decode($body->getContents()); $json = \Safe\json_decode($body->getContents());
if (!isset($json->mailboxes)) { if (!isset($json->mailboxes)) {
throw new MissingArgument('mailboxes', 'array', 'mailboxes names'); throw new MissingArgument('mailboxes', 'array', 'mailboxes ids');
} }
$output = [ $output = [
'mailboxes' => $json->mailboxes, 'mailboxes' => $json->mailboxes,
'total' => count($json->mailboxes), 'total' => count($json->mailboxes),
'unregistered' => 0 'unregistered' => [
'total' => 0,
'mailboxes' => []
]
]; ];
foreach ($json->mailboxes as $mailbox_name) { foreach ($json->mailboxes as $mailbox_id) {
if ($service->unregister($mailbox_name)) { $mailbox = $service->getRepository()->fetchById($mailbox_id);
$output['unregistered'] ++; if ($service->unregister($mailbox->getName())) {
$output['unregistered']['total'] ++;
$output['unregistered']['mailboxes'] []= [
'id' => $mailbox->getId(),
'name' => $mailbox->getName(),
'unregistered' => true
];
} }
} }
return $this->withJson($response, $output); return $this->withJson($response, $output);

View File

@ -0,0 +1,38 @@
<?php
namespace ProVM\Common\Middleware;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Server\RequestHandlerInterface;
use Psr\Log\LoggerInterface;
class Logging
{
public function __construct(LoggerInterface $logger) {
$this->setLogger($logger);
}
protected LoggerInterface $logger;
public function getLogger(): LoggerInterface
{
return $this->logger;
}
public function setLogger(LoggerInterface $logger): Logging
{
$this->logger = $logger;
return $this;
}
public function __invoke(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
{
$response = $handler->handle($request);
$output = [
'uri' => var_export($request->getUri(), true),
'body' => $request->getBody()->getContents()
];
$this->getLogger()->info(\Safe\json_encode($output, JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE));
return $response;
}
}

View File

@ -118,6 +118,7 @@ class Mailboxes extends Base
try { try {
$mailbox = $this->getRepository()->fetchByName($mailbox_name); $mailbox = $this->getRepository()->fetchByName($mailbox_name);
} catch (BlankResult $e) { } catch (BlankResult $e) {
$this->getLogger()->error($e);
// It's already unregistered // It's already unregistered
return true; return true;
} }

View File

@ -0,0 +1,2 @@
<?php
$app->add($app->getContainer()->get(ProVM\Common\Middleware\Logging::class));

View File

@ -18,4 +18,5 @@ return [
]); ]);
}, },
'attachments_folder' => $_ENV['ATTACHMENTS_FOLDER'], 'attachments_folder' => $_ENV['ATTACHMENTS_FOLDER'],
'logs_folder' => '/logs',
]; ];

View File

@ -7,5 +7,8 @@ return [
$container->get(Nyholm\Psr7\Factory\Psr17Factory::class), $container->get(Nyholm\Psr7\Factory\Psr17Factory::class),
$container->get(Psr\Log\LoggerInterface::class) $container->get(Psr\Log\LoggerInterface::class)
); );
},
ProVM\Common\Middleware\Logging::class => function(ContainerInterface $container) {
return new ProVM\Common\Middleware\Logging($container->get('request_logger'));
} }
]; ];

View File

@ -10,10 +10,20 @@ return [
$handler->setFormatter($container->get(Monolog\Formatter\SyslogFormatter::class)); $handler->setFormatter($container->get(Monolog\Formatter\SyslogFormatter::class));
return $handler; return $handler;
}, },
'request_logger' => function(ContainerInterface $container) {
$logger = new Monolog\Logger('request_logger');
$handler = new Monolog\Handler\RotatingFileHandler(implode(DIRECTORY_SEPARATOR, [$container->get('logs_folder'), 'requests.log']));
$handler->setFormatter($container->get(Monolog\Formatter\SyslogFormatter::class));
$dedupHandler = new Monolog\Handler\DeduplicationHandler($handler, null, Monolog\Level::Info);
$logger->pushHandler($dedupHandler);
$logger->pushProcessor($container->get(Monolog\Processor\PsrLogMessageProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\IntrospectionProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\MemoryUsageProcessor::class));
return $logger;
},
Psr\Log\LoggerInterface::class => function(ContainerInterface $container) { Psr\Log\LoggerInterface::class => function(ContainerInterface $container) {
$logger = new Monolog\Logger('file_logger'); $logger = new Monolog\Logger('file_logger');
$logger->pushHandler($container->get(Monolog\Handler\DeduplicationHandler::class)); $logger->pushHandler($container->get(Monolog\Handler\DeduplicationHandler::class));
//$logger->pushHandler($container->get(Monolog\Handler\RotatingFileHandler::class));
$logger->pushProcessor($container->get(Monolog\Processor\PsrLogMessageProcessor::class)); $logger->pushProcessor($container->get(Monolog\Processor\PsrLogMessageProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\IntrospectionProcessor::class)); $logger->pushProcessor($container->get(Monolog\Processor\IntrospectionProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\MemoryUsageProcessor::class)); $logger->pushProcessor($container->get(Monolog\Processor\MemoryUsageProcessor::class));

View File

@ -108,7 +108,12 @@ class Message implements Model
} }
public function getState(string $name): \ProVM\Emails\Model\State\Message public function getState(string $name): \ProVM\Emails\Model\State\Message
{ {
return $this->getStates()[$name]; try {
return $this->getStates()[$name];
} catch (\Exception $e) {
$this->newState($name);
return $this->getStates()[$name];
}
} }
public function addState(\ProVM\Emails\Model\State\Message $state): Message public function addState(\ProVM\Emails\Model\State\Message $state): Message
{ {
@ -127,6 +132,7 @@ class Message implements Model
$this->addState((new \ProVM\Emails\Model\State\Message()) $this->addState((new \ProVM\Emails\Model\State\Message())
->setName($name) ->setName($name)
->setMessage($this) ->setMessage($this)
->setValue(false)
); );
return $this; return $this;
} }
@ -143,31 +149,31 @@ class Message implements Model
{ {
return $this->getState('downloaded_attachments')->getValue() ?? false; return $this->getState('downloaded_attachments')->getValue() ?? false;
} }
public function hasScheduledDownloads(): bool
{
return $this->getState('scheduled_downloads')->getValue() ?? false;
}
public function doesHaveAttachments(): Message public function doesHaveAttachments(): Message
{ {
if (!isset($this->getState()['has_attachments'])) {
$this->newState('has_attachments');
}
$this->getState('has_attachments')->setValue(true); $this->getState('has_attachments')->setValue(true);
return $this; return $this;
} }
public function doesHaveValidAttachments(): Message public function doesHaveValidAttachments(): Message
{ {
if (!isset($this->getStates()['valid_attachments'])) {
$this->newState('valid_attachments');
}
$this->getState('valid_attachments')->setValue(true); $this->getState('valid_attachments')->setValue(true);
return $this; return $this;
} }
public function doesHaveDownloadedAttachments(): Message public function doesHaveDownloadedAttachments(): Message
{ {
if (!isset($this->getStates()['downloaded_attachments'])) {
$this->newState('downloaded_attachments');
}
$this->getState('downloaded_attachments')->setValue(true); $this->getState('downloaded_attachments')->setValue(true);
return $this; return $this;
} }
public function doesHaveScheduledDownloads(): Message
{
$this->getState('scheduled_downloads')->setValue(true);
return $this;
}
protected array $attachments; protected array $attachments;
public function getAttachments(): array public function getAttachments(): array
@ -205,7 +211,8 @@ class Message implements Model
'states' => [ 'states' => [
'has_attachments' => $this->hasAttachments(), 'has_attachments' => $this->hasAttachments(),
'valid_attachments' => $this->hasValidAttachments(), 'valid_attachments' => $this->hasValidAttachments(),
'downloaded_attachments' => $this->hasDownloadedAttachments() 'downloaded_attachments' => $this->hasDownloadedAttachments(),
'scheduled_downloads' => $this->hasScheduledDownloads()
], ],
'attachments' => $this->hasValidAttachments() ? array_map(function(Attachment $attachment) { 'attachments' => $this->hasValidAttachments() ? array_map(function(Attachment $attachment) {
return $attachment->toArray(); return $attachment->toArray();

View File

@ -111,7 +111,8 @@ class Message extends Repository
$valid_states = [ $valid_states = [
'has_attachments', 'has_attachments',
'valid_attachments', 'valid_attachments',
'downloaded_attachments' 'downloaded_attachments',
'scheduled_downloads'
]; ];
foreach ($valid_states as $state_name) { foreach ($valid_states as $state_name) {
try { try {

View File

@ -0,0 +1,38 @@
<?php
namespace ProVM\Common\Middleware;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Server\RequestHandlerInterface;
use Psr\Log\LoggerInterface;
class Logging
{
public function __construct(LoggerInterface $logger) {
$this->setLogger($logger);
}
protected LoggerInterface $logger;
public function getLogger(): LoggerInterface
{
return $this->logger;
}
public function setLogger(LoggerInterface $logger): Logging
{
$this->logger = $logger;
return $this;
}
public function __invoke(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
{
$response = $handler->handle($request);
$output = [
'uri' => var_export($request->getUri(), true),
'body' => $request->getBody()->getContents()
];
$this->getLogger()->info(\Safe\json_encode($output, JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE));
return $response;
}
}

View File

@ -4,6 +4,7 @@ $app = require_once implode(DIRECTORY_SEPARATOR, [
'setup', 'setup',
'app.php' 'app.php'
]); ]);
Monolog\ErrorHandler::register($app->getContainer()->get(Psr\Log\LoggerInterface::class));
try { try {
$app->run(); $app->run();
} catch (Error | Exception $e) { } catch (Error | Exception $e) {

View File

@ -0,0 +1,2 @@
<?php
$app->add($app->getContainer()->get(ProVM\Common\Middleware\Logging::class));

View File

@ -14,5 +14,6 @@ return [
$container->get('resources_folder'), $container->get('resources_folder'),
'commands' 'commands'
]); ]);
} },
'logs_folder' => '/logs',
]; ];

View File

@ -0,0 +1,8 @@
<?php
use Psr\Container\ContainerInterface;
return [
ProVM\Common\Middleware\Logging::class => function(ContainerInterface $container) {
return new ProVM\Common\Middleware\Logging($container->get('request_logger'));
}
];

View File

@ -2,14 +2,28 @@
use Psr\Container\ContainerInterface; use Psr\Container\ContainerInterface;
return [ return [
\Monolog\Handler\RotatingFileHandler::class => function(ContainerInterface $container) { Monolog\Handler\RotatingFileHandler::class => function(ContainerInterface $container) {
$handler = new \Monolog\Handler\RotatingFileHandler($container->get('log_file')); $handler = new Monolog\Handler\RotatingFileHandler($container->get('log_file'));
$handler->setFormatter($container->get(\Monolog\Formatter\LineFormatter::class)); $handler->setFormatter($container->get(Monolog\Formatter\LineFormatter::class));
return $handler; return $handler;
}, },
\Psr\Log\LoggerInterface::class => function(ContainerInterface $container) { 'request_logger' => function(ContainerInterface $container) {
$logger = new \Monolog\Logger('file_logger'); $logger = new Monolog\Logger('request_logger');
$logger->pushHandler($container->get(\Monolog\Handler\RotatingFileHandler::class)); $handler = new Monolog\Handler\RotatingFileHandler(implode(DIRECTORY_SEPARATOR, [$container->get('logs_folder'), 'requests.log']));
$handler->setFormatter($container->get(Monolog\Formatter\SyslogFormatter::class));
$dedupHandler = new Monolog\Handler\DeduplicationHandler($handler, null, Monolog\Level::Info);
$logger->pushHandler($dedupHandler);
$logger->pushProcessor($container->get(Monolog\Processor\PsrLogMessageProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\IntrospectionProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\MemoryUsageProcessor::class));
return $logger; return $logger;
} },
Psr\Log\LoggerInterface::class => function(ContainerInterface $container) {
$logger = new Monolog\Logger('file_logger');
$logger->pushHandler($container->get(Monolog\Handler\RotatingFileHandler::class));
$logger->pushProcessor($container->get(Monolog\Processor\PsrLogMessageProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\IntrospectionProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\MemoryUsageProcessor::class));
return $logger;
},
]; ];

View File

@ -0,0 +1,16 @@
<?php
namespace ProVM\Common\Controller;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use ProVM\Common\Service\Api as Service;
use Psr\Log\LoggerInterface;
class Api
{
public function __invoke(ServerRequestInterface $request, ResponseInterface $response, Service $service, LoggerInterface $logger): ResponseInterface
{
$json = $request->getParsedBody();
return $service->sendRequest($json);
}
}

View File

@ -0,0 +1,38 @@
<?php
namespace ProVM\Common\Middleware;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Server\RequestHandlerInterface;
use Psr\Log\LoggerInterface;
class Logging
{
public function __construct(LoggerInterface $logger) {
$this->setLogger($logger);
}
protected LoggerInterface $logger;
public function getLogger(): LoggerInterface
{
return $this->logger;
}
public function setLogger(LoggerInterface $logger): Logging
{
$this->logger = $logger;
return $this;
}
public function __invoke(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
{
$response = $handler->handle($request);
$output = [
'uri' => var_export($request->getUri(), true),
'body' => $request->getParsedBody()
];
$this->getLogger()->info(\Safe\json_encode($output, JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE));
return $response;
}
}

47
ui/common/Service/Api.php Normal file
View File

@ -0,0 +1,47 @@
<?php
namespace ProVM\Common\Service;
use Psr\Http\Client\ClientInterface;
use Psr\Http\Message\RequestFactoryInterface;
use Psr\Http\Message\ResponseInterface;
class Api
{
public function __construct(ClientInterface $client, RequestFactoryInterface $factory)
{
$this->setClient($client)
->setFactory($factory);
}
protected ClientInterface $client;
protected RequestFactoryInterface $factory;
public function getClient(): ClientInterface
{
return $this->client;
}
public function getFactory(): RequestFactoryInterface
{
return $this->factory;
}
public function setClient(ClientInterface $client): Api
{
$this->client = $client;
return $this;
}
public function setFactory(RequestFactoryInterface $factory): Api
{
$this->factory = $factory;
return $this;
}
public function sendRequest(array $request_data): ResponseInterface
{
$request = $this->getFactory()->createRequest(strtoupper($request_data['method']) ?? 'GET', $request_data['uri']);
if (strtolower($request_data['method']) !== 'get') {
$request->getBody()->write(\Safe\json_encode($request_data['data']));
$request->withHeader('Content-Type', 'application/json');
}
return $this->getClient()->sendRequest($request);
}
}

View File

@ -8,6 +8,7 @@ Monolog\ErrorHandler::register($app->getContainer()->get(Psr\Log\LoggerInterface
try { try {
$app->run(); $app->run();
} catch (Error | Exception $e) { } catch (Error | Exception $e) {
$app->getContainer()->get(Psr\Log\LoggerInterface::class)->debug(json_encode(compact('_REQUEST')));
$app->getContainer()->get(Psr\Log\LoggerInterface::class)->error($e); $app->getContainer()->get(Psr\Log\LoggerInterface::class)->error($e);
throw $e; throw $e;
} }

View File

@ -0,0 +1,4 @@
<?php
use ProVM\Common\Controller\Api;
$app->post('/api[/]', Api::class);

View File

@ -33,8 +33,8 @@
).append( ).append(
$('<button></button>').addClass('ui mini circular red icon button').append( $('<button></button>').addClass('ui mini circular red icon button').append(
$('<i></i>').addClass('remove icon') $('<i></i>').addClass('remove icon')
).click(() => { ).click(e => {
this.unregister() this.unregister($(e.currentTarget))
}) })
) )
} else { } else {
@ -42,7 +42,7 @@
register.append( register.append(
$('<button></button>').addClass('ui mini circular icon button').append( $('<button></button>').addClass('ui mini circular icon button').append(
$('<i></i>').addClass('save icon') $('<i></i>').addClass('save icon')
).click((e) => { ).click(e => {
this.register($(e.currentTarget)) this.register($(e.currentTarget))
}) })
) )
@ -72,20 +72,25 @@
}) })
}) })
} }
unregister() { unregister(button) {
const uri = '/mailboxes/unregister' const uri = '/mailboxes/unregister'
const data = { const data = {
mailboxes: [this.id] mailboxes: [this.id]
} }
button.html('').append(
$('<i></i>').addClass('redo loading icon')
)
return Send.delete({ return Send.delete({
uri, uri,
data data
}).then(response => { }).then(response => {
response.mailboxes.forEach(mb => { response.unregistered.mailboxes.forEach(mb => {
if (mb.id === this.id && mb.removed) { if (mb.id === this.id && mb.unregistered) {
this.id = null this.id = null
this.registered = false this.registered = false
mailboxes.draw().table() const tr = button.parent().parent()
tr.html('')
this.draw(tr)
} }
}) })
}) })

View File

@ -17,7 +17,8 @@
states = { states = {
attachments: false, attachments: false,
valid: false, valid: false,
downloaded: false downloaded: false,
scheduled: false
} }
attachments attachments
@ -81,10 +82,15 @@
const map_keys = { const map_keys = {
has_attachments: 'attachments', has_attachments: 'attachments',
valid_attachments: 'valid', valid_attachments: 'valid',
downloaded_attachments: 'downloaded' downloaded_attachments: 'downloaded',
scheduled_downloads: 'scheduled'
} }
Object.keys(states).forEach(k => { Object.keys(states).forEach(k => {
this.states[map_keys[k]] = states[k] if (k in map_keys) {
this.states[map_keys[k]] = states[k]
} else {
this.states[map_keys[k]] = false
}
}) })
return this return this
}, },
@ -103,6 +109,9 @@
}, },
downloaded: () => { downloaded: () => {
return this.states.downloaded return this.states.downloaded
},
scheduled: () => {
return this.states.scheduled
} }
} }
} }
@ -119,80 +128,99 @@
downloaded: () => { downloaded: () => {
this.states.downloaded = true this.states.downloaded = true
return this return this
},
scheduled: () => {
this.states.scheduled = true
return this
} }
} }
} }
draw() { draw() {
const format = Intl.DateTimeFormat('es-CL', {dateStyle: 'full', timeStyle: 'short'}) return {
let date = new Date() row: () => {
try { const format = Intl.DateTimeFormat('es-CL', {dateStyle: 'full', timeStyle: 'short'})
date = new Date(this.get().date()) let date = new Date()
} catch (e) { try {
console.debug(e) date = new Date(this.get().date())
} } catch (e) {
const valid = $('<td></td>').html($('<i></i>').addClass(this.has().valid() ? 'green check circle icon' : 'red times circle icon')) console.debug(e)
if (this.has().valid() && this.get().attachments().length > 0) {
const list = $('<div></div>').addClass('ui list')
this.get().attachments().forEach(attachment => {
list.append($('<div></div>').addClass('item').html(attachment.filename))
})
valid.append(
$('<div></div>').addClass('ui popup').append(list)
)
}
const tr = $('<tr></tr>').append(
$('<td></td>').html(this.get().subject())
).append(
$('<td></td>').html(format.format(date))
).append(
$('<td></td>').html(this.get().from().full)
).append(
$('<td></td>').html($('<i></i>').addClass(this.has().attachments() ? 'green check circle icon' : 'red times circle icon'))
).append(
valid
).append(
$('<td></td>').html($('<i></i>').addClass(this.has().downloaded() ? 'green check circle icon' : 'red times circle icon'))
).append(
$('<td></td>').append(
(this.has().attachments() && this.has().valid() && !this.has().downloaded()) ?
$('<button></button>').addClass('ui mini green circular icon button').append(
$('<i></i>').addClass('clock icon')
).click(() => {
this.download().attachments(this.get().id())
}) : ''
)
)
if (this.has().downloaded()) {
tr.find('td').each((i, td) => {
const content = $(td).html()
if (content.indexOf('icon') > -1) {
return
} }
$(td).html('') const valid = $('<td></td>').html($('<i></i>').addClass(this.has().valid() ? 'green check circle icon' : 'red times circle icon'))
$(td).append( if (this.has().valid() && this.get().attachments().length > 0) {
$('<a></a>').attr('href', '{{$urls->base}}/emails/message/' + this.id).html(content) const list = $('<div></div>').addClass('ui list')
this.get().attachments().forEach(attachment => {
list.append($('<div></div>').addClass('item').html(attachment.filename))
})
valid.append(
$('<div></div>').addClass('ui popup').append(list)
)
}
const tr = $('<tr></tr>').attr('data-id', this.get().id()).append(
$('<td></td>').html(this.get().subject())
).append(
$('<td></td>').html(format.format(date))
).append(
$('<td></td>').html(this.get().from().full)
).append(
$('<td></td>').html($('<i></i>').addClass(this.has().attachments() ? 'green check circle icon' : 'red times circle icon'))
).append(
valid
).append(
$('<td></td>').html($('<i></i>').addClass(this.has().downloaded() ? 'green check circle icon' : 'red times circle icon'))
).append(
$('<td></td>').append(
(this.has().attachments() && this.has().valid() && !this.has().downloaded()) ?
((this.has().scheduled()) ? this.draw().scheduledButton() : this.draw().scheduleButton()) : ''
)
) )
}) if (this.has().downloaded()) {
tr.find('td').each((i, td) => {
const content = $(td).html()
if (content.indexOf('icon') > -1) {
return
}
$(td).html('')
$(td).append(
$('<a></a>').attr('href', '{{$urls->base}}/emails/message/' + this.id).html(content)
)
})
}
return tr
},
scheduleButton: () => {
return $('<button></button>').addClass('ui mini circular icon button').append(
$('<i></i>').addClass('clock icon')
).click(e => {
this.download().attachments(e)
})
},
scheduledButton: () => {
return $('<i></i>').addClass('ui green circular inverted check icon')
},
schedulingButton: () => {
return $('<i></i>').addClass('ui circular inverted redo loading icon')
}
} }
return tr
} }
download() { download() {
return { return {
attachments: id => { attachments: event => {
const td = $(event.currentTarget).parent()
const id = this.get().id()
const uri = '/messages/schedule' const uri = '/messages/schedule'
const data = { const data = {
messages: [ messages: [
id id
] ]
} }
td.html('').append(this.draw().schedulingButton())
return Send.put({ return Send.put({
uri, uri,
data data
}).then(response => { }).then(response => {
if (response.scheduled > 0) { if (response.scheduled > 0) {
alert('Scheduled Attachments Job') td.html('').append(this.draw().scheduledButton())
} }
}) })
} }
@ -205,9 +233,6 @@
name: '', name: '',
count: '' count: ''
}, },
page: 1,
current: 0,
shown: 10,
total: null, total: null,
visible: [], visible: [],
get: function() { get: function() {
@ -221,7 +246,13 @@
}, },
messages: () => { messages: () => {
const uri = '/mailbox/{{$mailbox_id}}/messages/valid' const uri = '/mailbox/{{$mailbox_id}}/messages/valid'
return Send.get(uri).then(response => { return Send.get(uri).then((response, status, jqXHR) => {
if (parseInt(jqXHR.status/100) !== 2 || jqXHR.status === 204) {
$(this.id.results).html('').append(
this.draw().empty()
)
return
}
if (this.total === null) { if (this.total === null) {
$(this.id.count).html(' (' + response.total + ')') $(this.id.count).html(' (' + response.total + ')')
this.total = response.total this.total = response.total
@ -244,7 +275,6 @@
} }
}, },
draw: function() { draw: function() {
const columns = 8
return { return {
loader: () => { loader: () => {
$(this.id.results).html('').append( $(this.id.results).html('').append(
@ -264,8 +294,6 @@
this.draw().head() this.draw().head()
).append( ).append(
this.draw().body() this.draw().body()
).append(
this.draw().footer()
) )
}, },
head: () => { head: () => {
@ -280,43 +308,8 @@
} }
values.push(v) values.push(v)
} }
const dropdown = $('<div></div>').addClass('ui scrolling dropdown').append(
$('<i></i>').addClass('dropdown icon')
).append(
$('<div></div>').addClass('text').html(this.shown)
).dropdown({
values,
onChange: (value, text, $choice) => {
if (value === this.shown) {
return
}
this.shown = parseInt(value)
// Trigger pending transition
$choice.parent().parent().dropdown('hide')
this.get()
}
})
return $('<thead></thead>').append( return $('<thead></thead>').append(
$('<tr></tr>').append(
$('<th></th>').addClass('right aligned').attr('colspan', columns).append(
dropdown
).append(
$('<span></span>').addClass('ui basic label').html((this.current + 1) + ' - ' + Math.min(this.shown + this.current, this.total ?? 99999999))
).append(
$('<button></button>').addClass('ui mini circular icon button').append(
$('<i></i>').addClass('left chevron icon')
).click(() => {
this.prev()
})
).append(
$('<button></button>').addClass('ui mini circular icon button').append(
$('<i></i>').addClass('right chevron icon')
).click(() => {
this.next()
})
)
)
).append(
$('<tr></tr>').append( $('<tr></tr>').append(
$('<th></th>').html('#') $('<th></th>').html('#')
).append( ).append(
@ -339,7 +332,7 @@
body: () => { body: () => {
const tbody = $('<tbody></tbody>') const tbody = $('<tbody></tbody>')
this.visible.forEach((m, i) => { this.visible.forEach((m, i) => {
const row = m.draw() const row = m.draw().row()
row.prepend( row.prepend(
$('<td></td>').html(i + this.current + 1) $('<td></td>').html(i + this.current + 1)
) )
@ -347,94 +340,12 @@
}) })
return tbody return tbody
}, },
footer: () => {
const pages = this.lastPage()
const paging = $('<div></div>').addClass('ui right floated pagination menu')
if (this.page !== 1) {
paging.append(
$('<a></a>').addClass('icon item').append(
$('<i></i>').addClass('step backward icon')
).click(() => {
this.first()
})
)
}
for (let i = 0; i < pages; i ++) {
const page = $('<a></a>').addClass('item').html(i + 1).click(() => {
this.goto(i + 1)
})
if (i + 1 === this.page) {
page.addClass('active')
}
paging.append(page)
}
if (this.page !== this.lastPage()) {
paging.append(
$('<a></a>').addClass('icon item').append(
$('<i></i>').addClass('step forward icon')
).click(() => {
this.last()
})
)
}
return $('<tfoot></tfoot>').append(
$('<tr></tr>').append(
$('<th></th>').attr('colspan', columns).append(paging)
)
)
},
empty: () => { empty: () => {
$(this.id.count).html(' (0)')
return $('<div></div>').addClass('ui message').html('No messages found.') return $('<div></div>').addClass('ui message').html('No messages found.')
} }
} }
}, },
lastPage: function() {
let pages = Math.floor(this.total / this.shown)
if (this.total / this.shown > pages) {
pages ++
}
return pages
},
first: function() {
if (this.current === 0) {
return
}
this.page = 1
this.current = 0
this.get()
},
next: function() {
if (this.current + this.shown >= this.total) {
return
}
this.page ++
this.current += this.shown
this.get()
},
goto: function(page) {
if (this.page === page) {
return
}
this.page = page
this.current = (this.page - 1) * this.shown
this.get()
},
prev: function() {
if (this.current < this.shown) {
return
}
this.page --
this.current -= this.shown
this.get()
},
last: function() {
if (this.page === this.lastPage()) {
return
}
this.page = this.lastPage()
this.current = (this.page - 1) * this.shown
this.get()
},
setup: function() { setup: function() {
this.get().mailbox().then(() => { this.get().mailbox().then(() => {
this.get().messages() this.get().messages()

View File

@ -3,7 +3,7 @@
@section('emails_content') @section('emails_content')
<h3>Message - <span id="subject"></span> - <span id="from"></span></h3> <h3>Message - <span id="subject"></span> - <span id="from"></span></h3>
<h4 id="date_time"></h4> <h4 id="date_time"></h4>
<div class="ui list" id="attachments"></div> <div class="ui bulleted link list" id="attachments"></div>
@endsection @endsection
@push('page_scripts') @push('page_scripts')
@ -82,7 +82,7 @@
attachments: parent => { attachments: parent => {
this.get().attachments().forEach(attachment => { this.get().attachments().forEach(attachment => {
parent.append( parent.append(
$('<a></a>').attr('href', _urls.base + '/attachment/' + attachment.id).attr('download', attachment.fullname).html(attachment.fullname) $('<a></a>').attr('href', '{{$urls->base}}/attachment/' + attachment.id).attr('download', attachment.fullname).html(attachment.fullname)
) )
}) })
} }

View File

@ -14,7 +14,8 @@
const uri = '/mailboxes/registered' const uri = '/mailboxes/registered'
this.draw().loading() this.draw().loading()
return Send.get(uri).then((response, status, jqXHR) => { return Send.get(uri).then((response, status, jqXHR) => {
if (parseInt(jqXHR.status / 100) !== 2) { if (parseInt(jqXHR.status / 100) !== 2 || jqXHR.status === 204) {
this.draw().empty()
return return
} }
if (jqXHR.status === 200) { if (jqXHR.status === 200) {
@ -44,7 +45,7 @@
const parent = $(this.div_id) const parent = $(this.div_id)
parent.html('') parent.html('')
if (this.mailboxes.length === 0) { if (this.mailboxes.length === 0) {
parent.html('No mailboxes registered.') this.draw().empty()
return return
} }
const list = $('<div></div>').addClass('ui list') const list = $('<div></div>').addClass('ui list')
@ -54,6 +55,13 @@
) )
}) })
parent.append(list) parent.append(list)
},
empty: () => {
const parent = $(this.div_id)
parent.html('')
parent.append(
$('<div></div>').addClass('ui message').html('No mailboxes registered.')
)
} }
} }
} }

View File

@ -1,39 +1,33 @@
<script type="text/javascript"> <script type="text/javascript">
const Send = { const Send = {
base_url: '{{$urls->api}}', base_url: '{{$urls->api}}',
base: function({method, uri, data = null, dataType = 'json', contentType = 'application/json'}) { base: function({method, uri, data = null}) {
const url = this.base_url + '/' + uri.replace(/^\//g, '') const request = {
uri: uri,
method
}
const options = { const options = {
method, url: this.base_url,
url, method: 'post',
crossDomain: true,
headers: {
'Authorization': [
'Bearer {{$api_key}}'
]
},
dataType
} }
if (method.toLowerCase() !== 'get' && data !== null) { if (method.toLowerCase() !== 'get' && data !== null) {
if (!(typeof data === 'string' || data instanceof String)) { request['data'] = data
options['data'] = JSON.stringify(data)
}
options['contentType'] = contentType
} }
options['data'] = request
return $.ajax(options) return $.ajax(options)
}, },
get: function(uri, dataType = 'json') { get: function(uri) {
return this.base({method: 'get', uri, dataType}) return this.base({method: 'get', uri})
}, },
post: function({uri, data, dataType = 'json', contentType = 'application/json'}) { post: function({uri, data}) {
return this.base({method: 'post', uri, data, dataType, contentType}) return this.base({method: 'post', uri, data})
}, },
put: function({uri, data, dataType = 'json', contentType = 'application/json'}) { put: function({uri, data}) {
return this.base({method: 'put', uri, data, dataType, contentType}) return this.base({method: 'put', uri, data})
}, },
delete: function({uri, data, dataType = 'json', contentType = 'application/json'}) { delete: function({uri, data}) {
return this.base({method: 'delete', uri, data, dataType, contentType}) return this.base({method: 'delete', uri, data})
} }
} }
const _urls = JSON.parse('{!! Safe\json_encode($urls) !!}') //const _urls = JSON.parse('{!! Safe\json_encode($urls) !!}')
</script> </script>

View File

@ -0,0 +1,2 @@
<?php
$app->add($app->getContainer()->get(ProVM\Common\Middleware\Logging::class));

View File

@ -20,6 +20,7 @@ return [
$arr['base'], $arr['base'],
'cache' 'cache'
]); ]);
$arr['logs'] = '/logs';
return (object) $arr; return (object) $arr;
} }
]; ];

View File

@ -2,7 +2,12 @@
return [ return [
'urls' => function() { 'urls' => function() {
$arr = ['base' => $_SERVER['REQUEST_SCHEME'] . '://' . $_SERVER['HTTP_HOST']]; $arr = ['base' => $_SERVER['REQUEST_SCHEME'] . '://' . $_SERVER['HTTP_HOST']];
$arr['api'] = 'http://localhost:8080'; $arr['api'] = $_ENV['API_URI'];
return (object) $arr; return (object) $arr;
} },
'view_urls' => function() {
$arr = ['base' => $_SERVER['REQUEST_SCHEME'] . '://' . $_SERVER['HTTP_HOST']];
$arr['api'] = implode('/', [$arr['base'], 'api']);
return (object) $arr;
},
]; ];

View File

@ -4,12 +4,15 @@ use Psr\Container\ContainerInterface;
return [ return [
Psr\Http\Client\ClientInterface::class => function(ContainerInterface $container) { Psr\Http\Client\ClientInterface::class => function(ContainerInterface $container) {
return new GuzzleHttp\Client([ return new GuzzleHttp\Client([
'base_uri' => "http://proxy:8080", 'base_uri' => $container->get('urls')->api,
'headers' => [ 'headers' => [
'Authorization' => [ 'Authorization' => [
"Bearer {$container->get('api_key')}" "Bearer {$container->get('api_key')}"
] ]
] ]
]); ]);
} },
Psr\Http\Message\RequestFactoryInterface::class => function(ContainerInterface $container) {
return $container->get(Nyholm\Psr7\Factory\Psr17Factory::class);
},
]; ];

View File

@ -2,13 +2,13 @@
use Psr\Container\ContainerInterface; use Psr\Container\ContainerInterface;
return [ return [
\Slim\Views\Blade::class => function(ContainerInterface $container) { Slim\Views\Blade::class => function(ContainerInterface $container) {
return new \Slim\Views\Blade( return new Slim\Views\Blade(
$container->get('folders')->views, $container->get('folders')->views,
$container->get('folders')->cache, $container->get('folders')->cache,
null, null,
[ [
'urls' => $container->get('urls'), 'urls' => $container->get('view_urls'),
'api_key' => $container->get('api_key') 'api_key' => $container->get('api_key')
] ]
); );

View File

@ -0,0 +1,8 @@
<?php
use Psr\Container\ContainerInterface;
return [
ProVM\Common\Middleware\Logging::class => function(ContainerInterface $container) {
return new ProVM\Common\Middleware\Logging($container->get('request_logger'));
}
];

View File

@ -10,13 +10,23 @@ return [
$handler->setFormatter($container->get(Monolog\Formatter\SyslogFormatter::class)); $handler->setFormatter($container->get(Monolog\Formatter\SyslogFormatter::class));
return $handler; return $handler;
}, },
Psr\Log\LoggerInterface::class => function(ContainerInterface $container) { 'request_logger' => function(ContainerInterface $container) {
$logger = new Monolog\Logger('file_logger'); $logger = new Monolog\Logger('request_logger');
$logger->pushHandler($container->get(Monolog\Handler\DeduplicationHandler::class)); $handler = new Monolog\Handler\RotatingFileHandler(implode(DIRECTORY_SEPARATOR, [$container->get('folders')->logs, 'requests.log']));
//$logger->pushHandler($container->get(Monolog\Handler\RotatingFileHandler::class)); $handler->setFormatter($container->get(Monolog\Formatter\SyslogFormatter::class));
$dedupHandler = new Monolog\Handler\DeduplicationHandler($handler, null, Monolog\Level::Info);
$logger->pushHandler($dedupHandler);
$logger->pushProcessor($container->get(Monolog\Processor\PsrLogMessageProcessor::class)); $logger->pushProcessor($container->get(Monolog\Processor\PsrLogMessageProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\IntrospectionProcessor::class)); $logger->pushProcessor($container->get(Monolog\Processor\IntrospectionProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\MemoryUsageProcessor::class)); $logger->pushProcessor($container->get(Monolog\Processor\MemoryUsageProcessor::class));
return $logger; return $logger;
} },
Psr\Log\LoggerInterface::class => function(ContainerInterface $container) {
$logger = new Monolog\Logger('file_logger');
$logger->pushHandler($container->get(Monolog\Handler\DeduplicationHandler::class));
$logger->pushProcessor($container->get(Monolog\Processor\PsrLogMessageProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\IntrospectionProcessor::class));
$logger->pushProcessor($container->get(Monolog\Processor\MemoryUsageProcessor::class));
return $logger;
},
]; ];