Consol — Micro framework for command line PHP apps
Consol is a minimal and unopinionated micro framework for developing simple yet powerful command line apps and services that is easy to read and write, requires minimal prerequisite knowledge, has no dependencies and prefers common-case over edge-case solutions.
Hello, World!
// app.php
$app = new RichJenks\Consol\App;
$app->map('hello', function () {
return 'Hello World';
});
# terminal
php app.php hello
# Hello World
Key Features
- Routes terminal requests
- Simplifies parameters and options
- Generates command directory
- Packages your app as a PHAR
Requirements
- PHP >=5.5
- Phar PHP extension
phar.readonly = 0
in/etc/php5/cli/php.ini
Installation
Install via Composer with composer require richjenks/consol
or download and include it in your code with require 'consol/App.php';
Mapping Commands to Code
A Consol app could look like this:
#!/usr/local/bin/php
<?php require 'vendor/autoload.php';
$app->map('hello', function ($app) {
return 'Hello World';
}, 'Say "Hello" to the World');
$app->run();
Calling php app.php hello
from the command line will run this command. The map
function accepts 3 parameters:
- Command to trigger the route
- Callback to map to the command
- Optional description of the command
Commands can only contain alphanumerics, colons, hyphens and underscores.
Magic Commands
Magic Commands are commands that are not included in the command map and are called by special events. There are 2 Magic Commands and they are:
- Colon (
:
) which represents when no command is issued (the root command) - Question Mark (
?
) which represents an unknown command
Here’s an example of overriding the default unknown command handler:
$app->map('?', function ($app) {
return $app->say('I have no idea what I\'m doing...');
});
Both have default handlers which are overridden by mapping them to your own code. By default, the root command will show the Command Directory (which has its own section below) and unknown commands will display the text “Unknown Command” in red and may attempt to suggest an intended function in case of a typo via the suggest()
function, which is documented further down.
Using $app
The $app
object is made available to your commands and it exposes a number of useful functions:
Say
Outputs text to the console and should be used instead of echo
, print
, etc. because it outputs immediately (rather than at the end of the script) and allows responses to be edited by Middleware (which has its own section below). It accepts 2 parameters:
- The text to be output
- An optional color for the text
Available colors are aqua
, black
, blue
, cyan
, emerald
, gray
, lilac
, green
, maroon
, navy
, orange
, purple
, red
, silver
, white
and yellow
.
For example: $app->say('Oh hai!', 'green');
Color
Identical to say()
except it returns the text rather than outputting immediately.
Ask
Prompt the user to provide information, for example:
$name = ask('What is your name: ');
say("Hello, $name!");
It has three parameters but only the first is required:
- The text to display to the user
- Default value if user enters nothing (or
null
for no default value) - Regular Expression against which the value is checked
Providing
null
as the default value means that no default value will be considered — it does not mean it will returnnull
if no value is provided.
Here’s a more practical example:
$name = $app->ask('What is your name: ', null, '.+');
$app->say("Hello, $name!");
The above example will prompt the user for their name, but if none is given then it will ask again. The answer must be at least one character. ask
isn’t very clever, so you are free to define whatever logic you like.
Or to ask for confirmation:
$sure = $app->ask('Are you sure? [y/n] ', 'n', 'y|n');
$app->say(($sure === 'y') ? 'You are sure' : 'You are unsure');
Here are a few example regular expressions for common requirements:
.*
Accept anything, including empty values.+
Accept anything, must be at least one charactery|n
Accepts anything containing “y”, “n” (basic confirmation)^yes$|^no$|^maybe$
Accepts only “yes”, “no” or “maybe”
Neither validation rules nor default values are shown to the user — you are free to provide whatever information you wish in the first “question” parameter.
Progress
Shows a progress bar to the user and accepts two values: current progress and total. For example:
$app->say('Progressing...' . PHP_EOL);
$progress = 0;
while ($progress <= 70) {
$app->progress($progress, 70);
$progress += 3;
sleep(1);
}
$app->progress(70, 70);
$app->say(PHP_EOL . 'Progress complete' . PHP_EOL);
Table
### Suggest ### Get Request
This is a testing function that can be used to analyse the request. It returns an array with the keys command
, params
, options
and map
:
var_dump($app->get_request());
//
The param()
and option()
functions are preferred as they integrate will middleware and are generally “nicer” from a syntax perspective.
Command Directory
When mapping commands to callbacks, the $app->map()
function accepts an optional third argument: a description of the command. This is used to construct a directory of available commands that is shown by the default handler for the root command and it looks like this:
Commands:
hello Say "Hello" to the World
The directory makes use of the table()
function (documented below) to output tabular data and it can also be shown manually by returning $app->directory()
from within a command’s callback:
$app->map('directory', function ($app) {
$app->say($app->directory());
});
Parameters
Data can be retrieved from the command line in the form of parameters. For example, running php app.php hello World
will return Hello World
. $app->param()
can be used to get the value of a parameter in a one-indexed fashion — zero contains the command, which in this case is hello
.
For example, to say hello in the color of your choosing, you could register the following command:
$app->map('hello', function ($app) {
return $app->say(
'Hello ' . $app->param(1),
$app->param(2)
);
});
and then call:
php app.php hello World blue
# Outputs "Hello World" in blue
Options
Options are similar to Parameters but are prepended by two hyphens, e.g. --foo
, and can optionally provide values, e.g. --foo=bar
. They can be at any position in the command line call but are ignored by the indexing of parameters. For example, if you call php app.php foo --fizzbuzz bar
then foo
will still be parameter one and bar
will still be parameter two.
No options are sacred to a Consol app — for example, there is no concept of verbosity and no options are reserved, so you are free to implement any options you like!
The two types of options are Flag and Value options:
Flags
Flags are true
when present and false
when omitted:
$app->map('hello', function ($app) {
$greeting = 'Hello ' . $app->param(1);
if ($app->option('yell')) $greeting = strtoupper($greeting);
return $app->say($greeting);
});
php app.php hello World --yell
# Outputs "HELLO WORLD"
Flag options can also be passed in short form (e.g. -yv
) and you can check for the presence of multiple flags to facilitate this, e.g. if ($app->option('y', 'yell')) { ... }
. No link exists between short and long options so you are free to define any combination you wish by use of the option()
function.
Values
Values hold strings when present and are false
when omitted:
$app->map('hello', function ($app) {
$greeting = 'Hello ' . $app->param(1);
if (is_string($app->option('from')))
$greeting .= ', from ' . $app->option('from');
return $app->say($greeting);
});
php app.php hello World --from=Moon
# Outputs "Hello World, from Moon"
Note that Consol can’t distinguish between a flag option and an empty value option, so is_string()
should be used to check for the presence of a value option. Absent value options return false
like an omitted flag option.
Middleware
Middleware allows you to edit a request before it is routed through the map and the response output before it is shown to the console. Middleware in console applications is not as important as with web applications because of the proportion of things happening in the background and things visible to the user, but it is still a useful tool that allows multiple commands to be edited by the same filter(s).
Notes
Short params with values, e.g. -xvalue
are not supported and each character would be treated as a separate flag
Long options are strings if a value is provided or true if no value provided, but cannot be stacked
Short options are either true or false and are stackable
There is no relationship between short and long options (e.g. verbosity) so you are free to define behaviour yourself
Later options override earlier options
Parameters ignore option position and are parsed as if options are always at the end of the command
CLImate is used for all IO
Settings via $app->setting()
$app->var for custom variables, e.g. to pass data between commands