Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/willry/rabbitrun
RabbitRun - Mecanismo de fila com RabbitMQ
https://github.com/willry/rabbitrun
Last synced: 4 days ago
JSON representation
RabbitRun - Mecanismo de fila com RabbitMQ
- Host: GitHub
- URL: https://github.com/willry/rabbitrun
- Owner: WillRy
- Created: 2022-09-10T21:37:44.000Z (over 2 years ago)
- Default Branch: main
- Last Pushed: 2024-01-01T19:00:50.000Z (about 1 year ago)
- Last Synced: 2024-11-22T08:14:33.126Z (2 months ago)
- Language: PHP
- Size: 295 KB
- Stars: 0
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: readme.md
Awesome Lists containing this project
README
# RabbitRun - Mecanismo de fila
Qualidade de código:
[![Scrutinizer Code Quality](https://scrutinizer-ci.com/g/WillRy/rabbitrun/badges/quality-score.png?b=main)](https://scrutinizer-ci.com/g/WillRy/rabbitrun/?branch=main)
Mecanismo de fila utilizando a combinação do RabbitMQ e MySQL
- O RabbitMQ para entrega e distribuição da tarefa entre os workers
- O MySQL para conter detalhes sobre quais tarefas foram executadas e também quais devem ser ignoradas na filaCombinando o MySQL junto ao RabbitMQ, é possível pesquisar e excluir/ignorar itens da fila de forma simples, o que não é
possível somente com o RabbitMQ, devido as limitações dele em procurar itens dentro da fila**Workers**: É possível ter vários workers consumindo a fila ao mesmo tempo, pois o RabbitMQ trata a distribuição de
tarefa entre eles.## Graceful stop - parar somente após terminar task
Para garantir que os consumers/workers não reiniciem durante a execução de uma task, foram
implementados mecanismos que detectam os "signals" para encerrar o script e assim só finalizam
o script após terminar a task atual.Isso é feito através dos signals:
- SIGINT
- SIGTERMUsando os recursos:
- pcntl_sigprocmask(SIG_BLOCK, [SIGTERM, SIGINT]);
- pcntl_sigprocmask(SIG_UNBLOCK, [SIGTERM, SIGINT]);
- pcntl_signal_dispatch();## Requisitos
- MySQL
- PHP >= 8.0 com PDO
- RabbitMQ >= 3.8 (Filas do tipo Quorum são necessárias)## Demonstração
**Consumo de hardware baixo com 50k de mensagens e 40 workers**
![Painel administrativo](./midia/rabbitmq.png)
## Como publicar itens na fila
```php
$i,
"id_email" => $i,
"conteudo" => "blablabla"
];$worker->publish("queue_teste", $payload);
}```
## Como consumir itens da fila?
- Criar a classe de worker responsavel pelo processament **(Implementar a interface WorkerInterface)**
```php
onCheckStatus(function () {});
/**
* Executa ao pegar um item na fila
* Se retornar false, o item é descartado
*
* Se não retornar nada ou verdadeiro, o item é processado no método onExecuting
*/
$worker->onReceive(function ($dados) {
echo ' [x] [ receive ] ', json_encode($dados), "\n";
});/**
* Método que processa o item da fila
* É sempre necessária dar um destino a mensagem
*
* Fazer um $message->ack para marcar como "sucesso"
* Fazer um $message->nack() para descartar
* Fazer um $message->nack(true) para repor na fila
*
* Se alguma exception não for tratada, o item será recolocado
* na fila
*/
$worker->onExecuting(function (AMQPMessage $message, $dados) {echo ' [x] [ executing ] ', json_encode($dados), "\n";
sleep(1);
// $number = rand(0, 10) % 2 === 0;
// if ($number) throw new \Exception("Error");$message->ack();
echo ' [x] [ success ] ', json_encode($dados), "\n";
});/**
* Método que executa automaticamente caso aconteça uma exception não tratada
* durante o processamento
*/
$worker->onError(function (\Exception $e, $dados) {
echo ' [x] [ error ] ', json_encode($dados), "\n";
});$worker->consume("queue_teste");
```
## OBRIGATÓRIO
- Sempre execute um: **nack** ou **ack** para que a tarefa tenha um tratamento e não fique
infinito na fila.- nack(): marca como erro
- nack(true): marca como erro e coloca novamete na fila
- ack(): marca a tarefa como concluida## Demonstração
Dentro desse repositório tem a pasta **demo**, nela tem 3 exemplos:
- queue: consumer e publisher simples
- queue_db: consumer e publisher que mantém espelho/monitoramento da fila no banco de dados
- pubsub: consumer e publisher no modelo pubsub- **publisher.php**: Arquivo que publica itens na fila
- **consumer.php**: Arquivo que consome itens na fila, podendo ter várias instâncias## Exemplo com banco de dados
Na pasta **demo/queue_db** tem um exemplo que utiliza o RabbitMQ junto
do banco de dados, com o objetivo de:- Espelhar a fila no banco de dados (permite excluir um item na fila e ver a situação de cada uma)
- Monitorar o que os workers estão fazendo
- Poder pausar temporariamente a execução dos workers via bancoAo usar o PDO como driver de espelhamento de fila, deve ser criado as tabelas.
```sql
drop table if exists jobs;create table jobs
(
id bigint auto_increment primary key,
queue varchar(255) not NULL COMMENT 'queue name',
payload text not NULL COMMENT 'json content, filter with JSON_EXTRACT',
retries int not null default 0,
max_retries int not null default 10,
requeue_error boolean default true,
last_error text COMMENT 'last error description',
status_desc text COMMENT 'cancel/other status reason',
id_owner bigint COMMENT 'ID to determine the owner (user) of the item in the queue',
id_object bigint COMMENT 'ID to determine which object the queue item came from (ex: user, product and etc)r',
auto_delete_end boolean default false,
status enum('waiting','processing','canceled','error','success') default 'waiting',
start_at datetime,
end_at datetime,
INDEX idx_id_object (id_object),
INDEX idx_queue (queue),
INDEX idx_id_owner (id_owner)
);drop table if exists rabbit_monitor;
create table rabbit_monitor
(
id bigint primary key auto_increment,
name varchar(255) not null,
status int not null default 0,
jobID varchar(255) null,
groupName varchar(255) not null,
modifiedAt datetime null
);```