# Name Outthentic # Synopsis Generic testing, reporting, monitoring framework consuming [Outthentic::DSL](https://github.com/melezhik/outthentic-dsl). # Install cpanm Outthentic # Short introduction This is a quick tutorial on outthentic usage. ## Story being tested Story is just a perl script that yields something into stdout: $ cat story.pl print "I am OK\n"; print "I am outthentic\n"; Sometimes we can also call story file as scenario. ## Check file Story check is a bunch of lines stdout should match. Here we require to have \`I am OK' and \`I am outthentic' lines in stdout: $ cat story.check I am OK I am outthentic ## Story run Story run is process of verification of your story. A story verification is based on rules defined in story check file. The verification process consists of: * executing story file and saving stdout into file. * validating stdout against a story check. * returning result as the list of statuses, where every status relates to a single rule. See also [story runner](#story-runner). ## Suite A bunch of related stories is called project or suite. Sure you may have more then one story at your project. Just create a new directories with story files inside: $ mkdir hello $ echo 'print "hello"' > hello/story.pl $ echo hello > hello/story.check Now run the suite with `strun` command: $ strun ok 1 - perl /home/vagrant/projects/outthentic/examples/hello/story.pl succeeded ok 2 - stdout saved to /tmp/.outthentic/29566/QKDi3p573L ok 3 - output match 'hello' 1..3 ok /tmp/.outthentic/29566/home/vagrant/projects/outthentic/examples/hello/world/story.t .. ok 1 - perl /home/vagrant/projects/outthentic/examples/hello/world/story.pl succeeded ok 2 - stdout saved to /tmp/.outthentic/29566/xC3wrsS195 ok 3 - output match 'I am OK' ok 4 - output match 'I am outthentic' 1..4 ok All tests successful. Files=2, Tests=7, 0 wallclock secs ( 0.03 usr 0.00 sys + 0.09 cusr 0.01 csys = 0.13 CPU) Result: PASS # Calculator project example Here is more detailed tutorial where we will build a test suite for calculator program. Let's repeat it again - there are three basic outthentic entities: * project ( suite ) * story files ( scenarios ) * story checks ( rules ) ## Project Outthentic project is a bunch of related stories. Every project is _represented_ by a directory. Let's create a project to test a simple calculator application: $ mkdir calc-app $ cd calc-app ## Stories Stories are just perl scripts placed at project sub-directories and named `story.pl`. Every story is a small program with stdout gets tested. Let's create two stories for our calc project. One story for \`addition' operation and another for \`multiplication': # story directories $ mkdir addition # a+b $ mkdir multiplication # a*b # story files $ cat addition/story.pl use MyCalc; my $calc = MyCalc->new(); print $calc->add(2,2), "\n"; print $calc->add(3,3), "\n"; $ cat multiplication/story.pl use MyCalc; my $calc = MyCalc->new(); print $calc->mult(2,3), "\n"; print $calc->mult(3,4), "\n"; ## Story check files Story checks file contain validation rules for story.pl files. Every story.pl is always accompanied by story.check file. Story check files should be placed at the same directory as story.pl file. Lets add some rules for multiplication and addition stories: $ cat addition/story.check 4 6 $ cat multiplication/story.check 6 12 And finally lets run test suite: $ strun # Story term ambiguity Sometimes when we speak about _stories_ we mean an elementary scenario executed by story runner and represented by a couple of files - story.pl,story.check. In other cases we mean just a story.pl file or even story.check given separately. The one should always take _the context_ into account when talking about stories to avoid ambiguity. # Story runner Story runner - is a script to run outthentic stories. It is called `strun`. Runner consequentially goes several phases: ## A compilation phase. Stories are converted into perl test files *.t ( compilation phase ) and saved into temporary directory. ## An execution phase. [Prove](https://metacpan.org/pod/distribution/Test-Harness/bin/prove) utility recursively executes test files under temporary directory and thus provide final suite execution status. So after all outthentic project is just perl test project with *.t files inside, the difference is that while with common test project *.t files _are created by user_, in outthentic project *.t files _are generated_ by story files. # Story checks syntax Outthentic consumes [Outthentic DSL](https://github.com/melezhik/outthentic-dsl), so story checks are just rules defined in terms of Outthentic DSL - a language to validate unstructured text data. A few ( not all ) usage examples listed below. * plain strings checks Often all you need is to ensure that stdout has some strings in: # stdout HELLO HELLO WORLD 123456 # check list HELLO 123 # validation output OK - output matches 'HELLO' OK - output matches 'HELLO WORLD' OK - output matches '123' * regular expressions You may use regular expressions as well: # check list regexp: L+ regexp: \d # validation output OK - output matches /L+/ OK - output matches /\d/ See [check-expressions](https://github.com/melezhik/outthentic-dsl#check-expressions) in Outthentic::DSL documentation pages. * generators Yes you may generate new check entries on run time: # original check list Say HELLO # this generator creates 3 new check expressions: generator: [ qw{ say hello again } ] # final check list: Say HELLO say hello again See [generators](https://github.com/melezhik/outthentic-dsl#generators) in Outthentic::DSL documentation pages. * inline perl code What about inline arbitrary perl code? Well, it's easy! # check list regexp: number: (\d+) validator: [ ( capture()->[0] '>=' 0 ), 'got none zero number') ]; See [perl expressions](https://github.com/melezhik/outthentic-dsl#perl-expressions) in Outthentic::DSL documentation pages. * text blocks Need to validate that some lines goes successively? # stdout this string followed by that string followed by another one string with that string at the very end. # check list # this text block # consists of 5 strings # goes consequentially # line by line: begin: # plain strings this string followed by that string followed by another one # regexps patterns: regexp: with (this|that) # and the last one in a block at the very end end: See [comments-blank-lines-and-text-blocks](https://github.com/melezhik/outthentic-dsl#comments-blank-lines-and-text-blocks) in Outthentic::DSL documentation pages. # Hooks Story hooks are extension points to change [story run](#story-run) process. It's just files with perl code gets executed in the beginning of a story. You should name your hooks as `story.pm` and place them into story directory: $ cat addition/story.pm diag "hello, I am addition story hook"; sub is_number { [ 'regexp: ^\\d+$' ] } $ cat addition/story.check generator: is_number Reasons why you might need a hooks: * redefine story stdout * define generators * call downstream stories * other custom code # Hooks API Story hooks API provides several functions to hack into story run process: ## Redefine stdout *set_stdout(string)* Using set_stdout means that you never execute a story.pl to get a stdout, but instead you set stdout on your own side. This might be helpful when for some reasons you can't produce a stdout via story.pl file: This is simple an example : $ cat story.pm set_stdout("THIS IS I FAKE RESPONSE\n HELLO WORLD"); $ cat story.check THIS IS FAKE RESPONSE HELLO WORLD You may call `set_stdout()` more then once: set_stdout("HELLO WORLD"); set_stdout("HELLO WORLD2"); A final stdout will be: HELLO WORLD HELLO WORLD2 ## Upstream and downstream stories It is possible to run one story from another with the help of downstream stories. Downstream stories are reusable stories or modules. Story runner never executes downstream stories _directly_, instead of downstream story always gets called from the _upstream_ one: $ cat modules/create_calc_object/story.pl # this is a downstream story # to make story downstream # simply create story file # in modules/ directory use MyCalc; our $calc = MyCalc->new(); print ref($calc), "\n" $ cat modules/create_calc_object/story.check MyCalc $ cat addition/story.pl # this is a upstream story our $calc->addition(2,2); $ cat addition/story.pm # to run downstream story # call run_story function # inside upstream story hook # with a single parameter - story path, # note that you don't have to # leave modules/ directory in the path run_story( 'create_calc_object' ); $ cat multiplication/story.pl # this is a upstream story too our $calc->multiplication(2,2); $ cat multiplication/story.pm run_story( 'create_calc_object' ); Here are the brief comments to the example above: * to make story as downstream simply create story file at modules/ directory * call `run\_story(story\_path)` function inside upstream story hook to run downstream story. * you can call as many downstream stories as you wish. * you can call the same downstream story more than once. Here is an example code snippet: $ cat story.pm run_story( 'some_story' ) run_story( 'yet_another_story' ) run_story( 'some_story' ) * stories variables You may pass variables to downstream story with the second argument of `run_story()` function: run_story( 'create_calc_object', { use_floats => 1, use_complex_numbers => 1, foo => 'bar' } ) Story variables get accessed by `story_var()` function: $ cat create_calc_object/story.pm story_var('use_float'); story_var('use_complex_numbers'); story_var('foo'); * downstream stories may invoke other downstream stories * you can't use story variables in a none downstream story One word about sharing state between upstream/downstream stories. As downstream stories get executed in the same process as upstream one there is no magic about sharing data between upstream and downstream stories. The straightforward way to share state is to use global variables : # upstream story hook: our $state = [ 'this is upstream story' ] # downstream story hook: push our @$state, 'I was here' Of course more proper approaches for state sharing could be used as singeltones or something else. ## Story variables accessors There are some variables exposed to hooks API, they could be useful: * `project_root_dir()` - Root directory of outthentic project. * `test_root_dir()` - Test root directory. Root directory of generated perl test files , see also [story runner](#story-runner). * `config()` - Returns suite configuration hash object. See also [suite configuration](#suite-configuration). * `host()` - Returns value of \`--host' parameter. ## Ignore unsuccessful codes when run stories Every story is a perl script gets run by perl `system()` function returning an exit code. None zero exit codes result in test failures, this default behavior, to disable this say in hook file: $ cat story.pm ignore_story_err(1); ## PERL5LIB $project\_root\_directory/lib is added to PERL5LIB path, which make it easy to place custom modules under $project\_root\_directory'/lib directory: # my-app/lib/Foo/Bar/Baz.pm package Foo::Bar::Baz; ... # hook.pm use Foo::Bar::Baz; ... # Story runner client strun ## Options * `--root` Root directory of outthentic project. If root parameter is not set current working directory is assumed as project root directory. * `--debug` Enable/disable debug mode: * Increasing debug value results in more low level information appeared at output * Default value is 0, which means no debugging * Possible values: 0,1,2,3 * `--match_l` Truncate matching strings. In a TAP output truncate matching strings to {match_l} bytes; default value is 200. * `--story` Run only single story. This should be file path without extensions ( .pl, .check ): foo/story.pl foo/bar/story.pl bar/story.pl --story 'foo' # runs foo/ stories --story foo/story # runs foo/story.pl --story foo/bar/ # runs foo/bar/ stories * `--prove` Prove parameters. See [prove settings](#prove-settings) section for details. * `--host` This optional parameter sets base url or hostname of a service or application being tested. * `--ini` Configuration ini file path. See [suite configuration](#suite-configuration) section for details. * `--yaml` Yaml configuration file path. See [suite configuration](#suite-configuration) section for details. # Suite configuration Outthentic projects are configurable. Configuration data is passed via configuration files. There are two type of configuration files are supported: * .Ini style format * YAML format .Ini style configuration files are passed by `--ini` parameter $ strun --ini /etc/suites/foo.ini $ cat /etc/suites/foo.ini [main] foo = 1 bar = 2 There is no special magic behind ini files, except this should be [Config Tiny](https://metacpan.org/pod/Config::Tiny) compliant configuration file. Or you can choose YAML format for suite configuration by using `--yaml` parameter: $ strun --yaml /etc/suites/foo.yaml $ cat /etc/suites/foo.yaml main: foo : 1 bar : 2 Unless user sets path to configuration file explicitly by `--ini` or `--yaml` story runner looks for the files named suite.ini and _then_ ( if suite.ini is not found ) for suite.yaml at the current working directory. If configuration file is passed and read a related configuration data is accessible via config() function, for example in story hook file: $ cat story.pm my $foo = config()->{main}->{foo}; my $bar = config()->{main}->{bar}; # TAP Story runner emit results in a [TAP](https://testanything.org/) format. You may use your favorite TAP parser to port result to another test / reporting systems. Follow [TAP](https://testanything.org/) documentation to get more on this. Here is example for having output in JUNIT format: strun --prove "--formatter TAP::Formatter::JUnit" # Prove settings Story runner uses [prove utility](https://metacpan.org/pod/distribution/Test-Harness/bin/prove) to execute generated perl tests, you may pass prove related parameters using `--prove-opts`. Here are some examples: strun --prove "-Q" # don't show anythings unless test summary strun --prove "-q -s" # run prove tests in random and quite mode # Environment variables * `match_l` - In a suite runner output truncate matching strings to {match_l} bytes. See also `--match_l` in [options](#options). * `outth_show_story` - If set, then content of story.pl file gets dumped in TAP output. # Examples An example outthentic project lives at examples/ directory, to run it say this: $ strun --root examples/ # AUTHOR [Aleksei Melezhik](mailto:melezhik@gmail.com) # Home Page https://github.com/melezhik/outthentic # See also * [Sparrow](https://github.com/melezhik/sparrow) - outthentic suites manager. * [Outthentic::DSL](https://github.com/melezhik/outthentic-dsl) - Outthentic::DSL specification. * [Swat](https://github.com/melezhik/swat) - web testing framework consuming Outthentic::DSL. * Perl prove, TAP, Test::More # Thanks To God as the One Who inspires me to do my job!