Kint

Zero-setup replacement for var_dump().

Kint internals

Overview

Kint is typically used as an install-and-forget dumping tool but has plenty of options for customization and to be used as a library.

For both contributing to Kint and for customizing behavior a certain familiarity with Kint’s internals is required. This page provides a brief overview of what goes on under the hood.

First, a Kint\Value\Context\ContextInterface is passed into Kint\Parser along with a PHP value. The parser then creates a Kint\Value\AbstractValue to represent the value and returns it.

Once the AbstractValue has been built it’s passed to a Kint\Renderer\RendererInterface which produces a string output.

Creating the initial ContextInterface (Which includes the input value’s name among other things) and sticking all of this together is handled by the Kint\Kint facade (which is aliased to \Kint)

Parser

The parser receives an input value and a ContextInterface as parameters. It calls gettype() to get the first piece of information about the input data.

The parser loops through its Kint\Parser\PluginBeginInterface plugins and if one of them returns an AbstractValue this is the value returned.

If not, it switches on the type and calls one of various specialized parsing routines. In the case of arrays and objects this will result in recursive parsing of child values. Since references are a thing the parser has to detect recursive data structures to prevent infinite recursion.

After the basic parse is done and the AbstractValue has been created the parser loops through its PluginCompleteInterface plugins which can each alter the value and returns the final result.

Values

Values all must inherit from AbstractValue. All values have a string $type and an integer $flags which contains a bitmask of states common to all values (Such as when a value is blacklisted or reaches the depth limit)

Depending on the type of data a value may have more properties. A StringValue has a string, an encoding, and a length. Meanwhile an ArrayValue has a size and a list of child AbstractValue

Once the parser receives a value you no longer have information from the parent scope: what it was called, if it was a child of an array, if it was a private property, or if it was a reference. This is passed into the parser from the parent scope as a ContextInterface and stored in $context

Lastly, $representations stores a list of Kint\Value\Representation\RepresentationInterface attached to the value, which allows the rich renderer to have multiple tabs showing different representations of the same data.

Renderer

The renderer takes values produced by the parser and outputs a string. The exact process varies depending on the renderer. It could print out text or it could print out HTML. It could store the dumped data in a database or email it to the ISS.

RichRenderer

On its own the Kint\Renderer\RichRenderer only renders 3 types of representation: StringRepresentation, ValueRepresentation and ContainerRepresentation.

It has two sets of plugins: A list of Kint\Renderer\Rich\ValuePluginInterface for custom rendering of values, and a list of TabPluginInterface for custom rendering of representations in a value.

TextRenderer

The TextRenderer ignores representations entirely and provides the most succinct explanation possible. For scalar values it will show the value, for arrays it will show the children, and for objects it will show the “Children”.

An objects children are typically the properties but plugins can overwrite them in certain cases. For instance, if an object has no properties the IteratorPlugin will set the iterator contents to be the children.

TextRenderer has a list of Kint\Renderer\Text\PluginInterface which can alter the rendering of values, and a whitelist of parser plugins to stop all of the plugins from running on every dump.

The CliRenderer and PlainRenderer extend TextRenderer and provide additional features like colors.

Contributor guidelines

You can see a full list of contributor guidelines in the repository.

Since the parser takes PHP data by reference, and since input data can contain references itself, you can very easily break your application by accidentally altering data inside the dump. Be very careful not to alter user data.

Setting up a dev environment

For developing on core Kint you’ll need two things:

  1. Composer (PHP)
  2. Npm (NodeJS)

If you don’t have composer do yourself a favor and learn it. If you don’t have npm that’s understandable.

In any case, Kint uses composer and npm for development tools. Composer calls npm under the assumption that it’s in your $PATH.

If all of these are installed simply run composer install and all your dependencies will be installed automatically.

Format and Build

  • Static analysis can find bugs in code when even unit tests don’t. You should always run composer analyze before a commit and fix any errors.
  • Tests have to pass. You should always run ./vendor/bin/phpunit tests before a commit and fix any errors.
  • Since code style is a good thing you should always run composer format before a commit to fix any inconsistent code style.
  • Since we deliver reproducibly compiled JS/CSS/PHAR files in our repo you should always run composer build before a commit to rebuild the files and commit them.

If you fail any of these steps CI will complain.

Writing plugins »