Skip to content

Instantly share code, notes, and snippets.

@vielhuber
Last active February 7, 2025 14:20
Show Gist options
  • Save vielhuber/73f44b99823e7c863deea4ba901cd751 to your computer and use it in GitHub Desktop.
Save vielhuber/73f44b99823e7c863deea4ba901cd751 to your computer and use it in GitHub Desktop.
clean code best practices #php #laravel

prevent static classes

// bad
Example::calculate($obj)
  
// good
(new Example($obj))->calculate()

prevent static arguments

// bad
public static int $foo;

// good
public int $foo;

loop namings

$countries = ['Europe' => 'Germany'];

// bad
foreach($countries as $countries__key => $countries__value) { }

// good
foreach($countries as $continent => $country) { }

$dataToTest = [7, 42];

// bad
foreach($dataToTest as $dataToTest__value) { }

// good
foreach($dataToTest as $testValue) { }

prevent abbreviations

// bad
$model->calc()
  
// good
$model->calculate()

cases

// camel case
$variableFooBar
function fooBarBaz() {}
  
// pascal case
class ExampleClass {}

// snake case
$variable_in_template
Foo::first()->variable_from_db

prevent arrays

// bad
exampleFunction(['foo' => 'bar', 'bar' => 'baz'])
  
// good
exampleFunction(foo: 'bar', bar: 'baz')

prevent arrays (2)

// bad
$foo = (object) ['foo' => 'bar', 'bar' => 'baz']; // anonymous object with properties 'foo' and 'bar'

// good
$foo = (new Foo())->foo('bar')->bar('baz')->get(); // Foo object with properties 'foo' and 'bar'

use classes / use getters / prevent arrays

// bad
$mapping = getMapping('test'); // ['foo' => 42, 'bar' => 7]

// good
$mapping = (new MappingGetter())->value('test')->get(); // Mapping object
class Mapping {
    public ?int $foo; 
    public ?int $bar;
}
class MappingGetter {
    public string $value;
    public function value($value) { $this->value = $value; return $this; }
    public function get() {
      $mapping = (new Mapping());
      $mapping->foo = 42;
      $mapping->bar = 7;
      /* ... more logic ... */
      return $mapping;
    }
}

prevent monster classes / outsource in separate classes

// bad
$objectInBigModel->calculateSpecialValue();


// good
(new SpecialValueCalculator($objectInBigModel))->calculate()

use dtos

  • DTOs (Data Transfer Objects) are little "packages" that hold data meant to be moved around
  • the data is consistent, structured and typed
  • they improve IDE support
// bad
$userData = [
    'id' => 42,
    'name' => 'David',
    'email' => '[email protected]'
];
print_r($userData['id']); // 42

// best (php >= 8.1)
$userData = UserDTO::create(
    42,
    'David',
    '[email protected]'
);
print_r($userData->id); // 42
class UserDTO {
    private function __construct(
        public readonly int $id,
        public readonly string $name,
        public readonly string $email
    ) {}

    public static function create(int $id, string $name, string $email): UserDTO {
        // potentially modify data before storing
        // ...
        return new self($id, $name, $email);
    }
}

// good (php <8.1)
$userData = UserDTO::create(
    42,
    'David',
    '[email protected]'
);
print_r($userData->id()); // 42
class UserDTO {
    private int $id;
    private string $name;
    private string $email;
    private function __construct(int $id, string $name, string $email) {
        $this->id = $id;
        $this->name = $name;
        $this->email = $email;
    }
    public static function create(int $id, string $name, string $email): UserDTO {
        // potentially modify data before storing
        // ...
        return new self($id, $name, $email);
    }
    public function id(): int {
        return $this->id;
    }
    public function name(): string {
        return $this->name;
    }
    public function email(): string {
        return $this->email;
    }
}

use vos

  • VOs (value objects) represent one or multiple values
  • Cannot be changed once instantiated
  • Validate the value when instantiating
  • Have methods that convert, compare or format the value
  • DTOs and VOs serve different purposes and actually complement each other very well
    • DTOs transfer data (carry data between layers or systems, without holding much logic or responsibility)
    • VOs ensure the integrity of data (focus on modeling concepts within your domain)
// bad
$price = (float) 42;
$price = '$'.number_format($price/100, 2);
echo $price;

// good
$price = Price::fromInteger(42);
$price->formattedWithSymbol();
$price->value();
final readonly class Price
{
  public function __construct(private int $value) {
    if ($value < 0) {
      throw new PriceCannotBeBelowZero();
    }
  } 
  public static fromInteger(int $value): self {
    return self($value);
  } 
  private function convertToDollars(): float {
    return $this->value / 100;
  } 
  public function formattedWithSymbol(): string {
    return '$' . number_format($this->convertToDollars(), 2);
  } 
  public function value(): int {
    return $this->value;
  }
}

function chaining

// bad
$value = $foo->calc(save: true);

// good
$foo->save()->calc()->getValue();

// bad
(new Foo(data: 'bar'))->calc()
  
// good
(new Foo())->setData(data: 'bar')->calc()

use framework functions

// bad
$str = trim($str);
$str = mb_strtoupper($str);
$str = str_replace(' ', '', $str);

// good
$str = Str::of($str)->trim()->upper()->replace(' ', '')->toString();

prevent duplicate database calls

// bad
$model = Model::orderBy('id');
$count = $model->count(); // counts in db
$model = $model->get();

// good
$model = Model::orderBy('id');
$model = $model->get();
$count = $model->count(); // counts in collection

output console commands

// bad
echo $msg . PHP_EOL; // with newline in command
echo $msg . ' '; // without newline in command
echo str_pad($msg, 99, ' ', STR_PAD_RIGHT) . "\r"; // with overwrite in command
(new \Symfony\Component\Console\Output\ConsoleOutput())->writeln($msg); // with newline in controller
(new \Symfony\Component\Console\Output\ConsoleOutput())->info($msg); // without newline in controller
(new \Symfony\Component\Console\Output\ConsoleOutput())->write(str_pad($msg, 99, ' ', STR_PAD_RIGHT) . "\r"); // with overwrite in controller

// good
$this->info($msg); // with newline in command
$this->output->write($msg); // without newline in command
$this->output->write(str_pad($msg, 99, ' ', STR_PAD_RIGHT) . "\r"); // with overwrite in command
ConsoleOutputHelper::info($msg); // with newline in controller
ConsoleOutputHelper::write($msg); // without newline in controller
ConsoleOutputHelper::overwrite($msg); // with overwrite in controller

prevent too much nesting

// bad
function log($msg) : void {
  if(...) {
    echo $msg;
  }
}

// good
function log($msg) {
  if(!...) {
    return; 
  }
  echo $msg;
}

prevent too much nesting (2)

// bad
if($a === true) {
    if($b === true) {
        if($c === true) {
            return 42;
        }
    }
}
return 7;

// good
if($a !== true) { return 7; }
if($b !== true) { return 7; }
if($c !== true) { return 7; }
return 42;

prevent else

// bad
if($foo) {

}
elseif($bar) {

}

// good
if($foo) {

}
if($bar && !$foo) {

}

prevent too much function calls

// bad
foo($value, true, true, false, 42)
  
// good
foo(value: $value, save: true, cache: true, force: false, amount: 42)
$foo->withSave()->withCache()->withoutForce()->withAmount(42)->calculate($value);

outsource long code parts

// bad
$isSpecial = false;
if($foo) { $isSpecial = true; }
if($bar && !$foo) { $isSpecial = false; }
if($isSpecial) {
  //...
}

// good
function isSpecial() {
  if($foo) { return true; }
  if($bar) { return false; }
  return false;
}
if($this->isSpecial()) {
  //...
}

use carbon instead of strings

// bad
date('Y')-10

// good
Carbon::now()->subYears(10)->year

use "final" keyword

// bad
class NonDerivativeClass {}

// good
final class NonDerivativeClass {}

use php type hinting

// bad
public static function foo($a, $b = null, $c = false) {}

// good
protected function foo(int $a, ?string $b = null, bool $c = false): void {}

// bad
public $foo;

// good
protected ?int $foo;

use jobs to dispatch long running tasks

// bad
longRunningTask();

// good
CalculateLongRunningTaskQueue::dispatch();

multiline comments

// bad
// this is
// a multiline
// comment

// bad
/*
this is
a multiline
comment
*/

// good
/**
 * this is
 * a multiline
 * comment
 */

comment functions (e.g. using copilot)

// bad
public function foo(): ?float {}

// good
/**
 * Calculate the value for our example.
 *
 * @return float|null
 */
public function foo(): ?float {}

phpunit tests: the expected value should come first as an argument

// bad
$this->assertSame($foo, 'bar');

// good
$this->assertSame('bar', $foo);

always write tests (tdd), use factories

// good
Model::factory()->count(10)->create(['birth_date' => '1980-01-01']);
Model::factory()->count(10)->create(['birth_date' => '1990-01-01']);
Model::factory()->count(10)->create(['birth_date' => '2000-01-01']);
$this->assertEquals(30, Model::whereBornAfter(1970)->count());
$this->assertEquals(30, Model::whereBornAfter(1980)->count());
$this->assertEquals(20, Model::whereBornAfter(1990)->count());

always catch the most specific exception

// bad
catch(\Throwable $e) {
    print_r($e->getMessage());
}

// good
catch(\PDOException $e) {
    print_r($e->getMessage());
}

use enums

// bad
class Environment
{
    static function get($str) {
        if( $str === 'Entwicklung' ) { return 'dev'; }
        if( $str === 'Test' ) { return 'stage'; }
        if( $str === 'Produktion' ) { return 'prod'; }
    }
}
echo Environment::get('Produktion'); // "prod"

// good
enum Environment: string
{
    case Entwicklung = 'dev';
    case Test = 'stage';
    case Produktion = 'prod';
}
echo Environment::Produktion->value; // "prod"
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment