• Stars
    star
    167
  • Rank 225,287 (Top 5 %)
  • Language
    PHP
  • License
    MIT License
  • Created about 3 years ago
  • Updated over 1 year ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Ensure your Laravel applications keep a normal pulse

Laravel Defibrillator

Ensure your Laravel applications keep a normal rhythm

Latest Version on Packagist GitHub Tests Action Status GitHub Code Style Action Status Total Downloads

Laravel Defibrillator helps you ensure that aspects of your application that should be running at a regular interval are doing so.

Installation

You can install the package via composer:

composer require dyrynda/laravel-defibrillator

You can publish the config file with:

php artisan vendor:publish --provider="Dyrynda\Defibrillator\DefibrillatorServiceProvider" --tag="laravel-defibrillator-config"

Usage

When an abnormal heartbeat rhythm is detected, you can defibrillate your heart to get back to normalcy.

<?php

if ($this->hasAbnormalRhythm()) {
  $this->defibrillate();

  return;
}

What on Earth?

Consider a scheduled task that communicates with your application users on a regular interval.

This scheduled task queues notifications to your users based on some condition within your application.

In a normal situation, there is a handful of notifications to go out, and they are dispatched with in a few seconds.

But an application error causes your application to spiral out of control.

Queued notifications back up, your database is not being updated to flag notifications as having been sent, your error tracker floods with exceptions.

And then your scheduled task runs again.

Suddenly your queue has tens of thousands of pending jobs in it and you're stuck in a cycle that you can't keep up with.

Enter Laravel Defibrillator

The Laravel Defibrillator helps you keep track of individual components of your application that are expected to be called on a regular interval.

On each execution, you call the defibrillate() method to update a cache value, setting an acceptable rhythm.

For example;

<?php

// app/Console/Kernel.php
$schedule->job(NotifyUsers::class)->everyMinute();

// app/Jobs/NotifyUsers.php
public function handle()
{
  if ($this->hasAbnormalRhythm()) {
    $this->defibrillate();

    return;
  }

  // Regular processing
  $this->defibrillate();
}

By default, calling defibrillate() will put an item into your configured cache with a Carbon instance 90 seconds into the future. The cache key is the basename of the calling class. i.e. the cache key for the App\Jobs\NotifyUsers class will be NotifyUsers. You can override the heart() method in your class if you wish to have more control over this.

Within the realm of normal operation, your scheduled task will execute every 60 seconds and push the cached value another 90 seconds.

However, should your database become overwhelmed, or your queues full of backlogged email notifications, and your scheduled task misses an execution and the cached value is in the past, instead of adding further strain to the database or pushing more notifications on to the queue, the Defibrillator will instead push the cache value out another 90 seconds.

In doing so, you give your database a chance to keep up with the queue backlog without manual intervention.

You can even prevent lagging notifications from being sent should the hearbeat enter an abnormal rhythm between when the scheduled task ran and the notification is trying to be sent:

<?php
// app/Notifications/CustomerNotification.php
public function shouldSend(): bool
{
  return Cache::get('NotifyUsers')?->isFuture() ?? false;
}

How will I know this is happening?

It is beyond the scope of this package, however, you might consider conditionally dispatching a notification if you breach a threshold in abnormal rhythm.

<?php

// app/Jobs/NotifyUsers.php
public function handle()
{
  if ($this->hasAbnormalRhythm()) {
    $this->defibrillate();
+
+   CardiacEvent::dispatchIf(Cache::increment("{$this->heart()}:skipped") === 3);

    return;
  }

  // Regular processing
  $this->defibrillate();
+
+ Cache::forget("{$this->heart()}:skipped");
}

This way, if you have 3 consecutive defibrillations, you can dispatch an email, SMS, Slack, whatever notification to get on the case!

Alternatively, you might consider a scheduled task monitoring solution such as thenping.me.

Configuration

You can define the default interval of a normal heartbeat by setting the defibrillator.interval config value or override it per-class by adding an interval method where you are using the Defibrillator trait:

<?php

use Dyrynda\Defibrillator\Defibrillator;

class Artisan
{
  use Defibrillator;

  public function interval(): int
  {
    return 30;
  }
}

Support development

If you would like to support the on going maintenance and development of this package, please consider sponsoring me on GitHub.

Testing

composer test

Changelog

Please see CHANGELOG for more information on what has changed recently.

Contributing

Please see CONTRIBUTING for details.

Security Vulnerabilities

Please review our security policy on how to report security vulnerabilities.

Credits

License

The MIT License (MIT). Please see License File for more information.

More Repositories

1

laravel-cascade-soft-deletes

Cascading deletes for Eloquent models that implement soft deletes
PHP
935
star
2

laravel-model-uuid

This package allows you to easily work with UUIDs in your Laravel models
PHP
449
star
3

laravel-efficient-uuid

PHP
305
star
4

phpstorm-laravel-code-style

PhpStorm code style to meet Laravel's contribution guidelines
293
star
5

laravel-make-user

A simple Artisan command to create application users
PHP
109
star
6

laravel-nullable-fields

Handles saving empty fields as null for Eloquent models
PHP
104
star
7

dotfiles

Lua
57
star
8

nomad

Laravel-style database migrations wherever they may roam
PHP
56
star
9

vagabond

Eloquent without a home
PHP
48
star
10

founder

Laravel starter application
PHP
48
star
11

carbon.vim

A Vim port of the IntelliJ Carbon theme
Vim Script
41
star
12

Laravel.tmTheme

A Sublime Text 3 theme based on the Laravel documentation colour scheme
40
star
13

founder-style

Founder repository style guide
17
star
14

laravel-owns-models

PHP
15
star
15

readme-generator

Generate README files that rock!
HTML
15
star
16

laravel-google-cse

Laravel wrapper for the Google Custom Search Engine API
PHP
9
star
17

laravel-ldap

Leverage Laravel's middleware to ensure your LDAP-authenticated users stay that way
PHP
6
star
18

git-workflow

A git workflow for facilitating client-facing User Acceptance Testing
6
star
19

RestClient

A REST API Client
PHP
4
star
20

laracasts-saving-json-into-database

Tests and implementation for forum post
JavaScript
4
star
21

terminable-jobs

PHP
3
star
22

telstra-sms

PHP
2
star
23

dyryme

dyry.me link shortener
PHP
2
star
24

amber-electric-php

PHP
2
star
25

simple-menu

PHP
1
star
26

zsh-completions

zsh completion scripts
Shell
1
star
27

chalrynda-blog

Michael & Rhiannon's travel blog
CSS
1
star
28

foods-for-life

Foods For Life Bedrock-based WordPress site
PHP
1
star
29

devdojo-events

PHP
1
star
30

deckset

1
star
31

bbc-good-food

PHP
1
star
32

hemp

PHP
1
star
33

s3upload

Code for blog series on uploading files to Amazon S3 from the browser
PHP
1
star