Git Product home page Git Product logo

popo-factory's Introduction

Making it easy to mock your POPO's / Value Objects

Latest Version on Packagist GitHub Tests Action Status Total Downloads

This package supports mocking POPOs or Value Objects, it inspects your class properties and populates them with random data generated by Faker. You can create class-based mock factories to have a fluent interface to mock POPOs for different states.

This is a fork of Data Transfer Object Factory.

Installation

You can install the package via composer:

composer require morrislaptop/popo-factory --dev

Usage

use Morrislaptop\PopoFactory\PopoFactory;

class PersonData
{
    public function __construct(
        public $firstName,
        public string $lastName,
        public string $email,
        public string $homeAddress,
        public ?string $companyName,
        public string $workAddress,
        public Carbon $dob,
        public PersonData $spouse,
    ) {
    }
}

PopoFactory::new(PersonData::class)->make();

Which creates an object that looks like this:

{
  "firstName": "Jada",
  "lastName": "Bechtelar",
  "email": "[email protected]",
  "homeAddress": "640 Feest Landing\nBernierburgh, PA 99277",
  "companyName": "Ratke Inc",
  "workAddress": "75107 Konopelski Radial\nRutherfordport, AK 22994",
  "dob": "2021-01-26T11:04:31.393991Z",
  "spouse": {
    "firstName": "Leola",
    "lastName": "Koss",
    "email": "[email protected]",
    "homeAddress": "852 Fabian Mills\nNorth Ward, NM 54459",
    "companyName": "Fahey and Sons",
    "workAddress": "671 Creola Prairie Apt. 663\nNorth Gretchenview, OR 75622-4176"
  }
}

The new method returns an instance of Morrislaptop\PopoFactory\PopoFactory which provides the following methods.

  • count() - Allows you to specify how many POPOs to be generated. They will be returned in an array.
  • make() - Called when you are ready to generate the POPO(s). Returns the generated object(s).
  • random() - Generates a random number of POPOs
  • sequence() - Alternates a specific state. (See below)
  • state() - Manually sets properties based on the array of values passed.

Examples of these methods can be found below.

// Creates two DTOs in an array
PopoFactory::new(PersonData::class)->count(2)->make();

// Sets the first name of every person to "Jim"
PopoFactory::new(PersonData::class)
    ->random()
    ->state([
        'firstName' => 'Jim',
    ])
    ->make();

// Alternates the names of each person between "Jim" and "Susie"
PopoFactory::new(PersonData::class)
    ->random()
    ->sequence(
        [ 'firstName' => 'Jim' ],
        [ 'firstName' => 'Susie' ]
    )
    ->make();

Creating Class Based Factories

It's useful to define specific factories for particular objects, which can easily be done by extending the PopoFactory class.

My specifying a typehint for the make() method you will also get typehints in your IDE for your mocked object.

/**
 * @method PersonData make
 */
class PersonDataFactory extends PopoFactory
{
    public static function factory(): static
    {
        return static::new(PersonData::class)->state([
            'firstName' => 'Craig'
        ]);
    }

    public function gotNoJob() {
        return $this->state([
            'companyName' => null,
        ]);
    }

    public function worksAtHome() {
        return $this->state(function ($attributes) {
            return [
                'workAddress' => $attributes['homeAddress']
            ];
        });
    }
}

Then using it in tests like so:

$person = PersonDataFactory::factory()
            ->gotNoJob()
            ->worksAtHome()
            ->make();

Extending

You can easily extend the factory to support other data types. You can do this through the static registerProvider() method on the PropertyFactory class. This method takes two arguments. The first should be the FQDN of the class you are providing (e.g. Carbon\Carbon) OR the built-in type (e.g. string). The second should be a callback that returns the generated value. This callback is passed two properties when called to assist in generating the value. The first is an instance of Anteris\FakerMap\FakerMap which can be used to help generate fake data. The second is an instance of ReflectionProperty which contains information about the property being generated.

For example, to support Carbon:

use Morrislaptop\PopoFactory\PropertyFactory;

use Anteris\FakerMap\FakerMap;

PropertyFactory::registerProvider('Carbon\Carbon', fn(FakerMap $fakerMap) => Carbon::parse(
    $fakerMap->closest('dateTime')->fake()
));

Plug

Want an easy way to persist your POPOs in Laravel? Check out laravel-popo-caster

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.

popo-factory's People

Contributors

erikgaal avatar lukio3 avatar morrislaptop avatar tigitz avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar

popo-factory's Issues

Multple errors in PHP 7.4

You're using the static return type and union type both features are only valid in PHP 8.

Here're some of the errors I'm getting.

ParseError : syntax error, unexpected '|', expecting ';' or '{'
 /home/albo/projects/popo-factory/src/PopoFactory.php:110
 /home/albo/projects/popo-factory/tests/PopoFactoryTest.php:48 

ParseError : syntax error, unexpected 'static' (T_STATIC)
 /home/albo/projects/popo-factory/tests/Popos/PersonDataFactory.php:12
 /home/albo/projects/popo-factory/tests/PopoFactoryTest.php:63

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.