NAME Sub::Middler - Middleware subroutine chaining SYNOPSIS use strict; use warnings; use Sub::Middler; my $middler=Sub::Middler->new; $middler->register(mw1(x=>1)); $middler->register(mw2(y=>10)); my $head=$middler->link( sub { print "Result: $_[0]\n"; } ); $head->(0); # Call the Chain # Middleware 1 sub mw1 { my %options=@_; sub { my ($next,$index)=@_; sub { my $work=$_[0]+$options{x}; $next->($work); } } } # Middleware 2 sub mw2 { my %options=@_; sub { my ($next, $index)=@_; sub { my $work= $_[0]*$options{y}; $next->( $work); } } } DESCRIPTION A small module, facilitating linking together subroutines, acting as middleware or filters into chains for flexible usage and low overhead runtime performance. To achieve these desirable attributes, the 'complexity' is offloaded in the definition of middleware/filters. They have to be wrapped in subroutines appropriately to facilitate the lexical binding. This differs from other 'sub chaining' modules as it does not use a loop internally to iterate of over a list of subroutines at runtime. As such there is no implicit call to the next item in the chain, each stage can run synchronously or asynchronously or even not at all. Each element in the chain is responsible for calling the next. Finally the arguments and signatures used to each stage of middleware are completely user defined. This allows reuse of the @_ array in calling subsequent segments for ultimate performance if you know what you're doing. API Managing a chain new my $object=Sub::Middler->new; Creates a empty middler object ready to accept middleware. The object is a blessed array reference which stores the middlewares directly. register $object->register(my_middlware()); Appends the middleware to the internal list for later linking. link $object->link($last); Links together the registered middleware stored internally. Each middleware is intrinsically linked to the next middlware in the list. The last middleware being linked to the $last argument, which must be a code ref. The $last code ref does not have to be strictly be middleware. Calls "die" if $last is not a code ref. Creating Middleware To achieve low over head in linking middleware, functional programming techniques (higher order functions). This also give the greatest flexibility to the middleware, as signatures are completely user defined. The trade off is that the middleware must be defined in a certain code structure. While this isn't difficult, it takes a minute to wrap your head around. Middlware definition Middleware must be a subroutine (top/name) which returns a anonymous subroutine (maker), which also returns a anonymous subroutine to perform work (kernel). This sounds complicated by this is what is looks like in code: sub my_middleware { (1) Top/name subroutine my %options=@_; Store any config sub { (2) maker sub is returned my ($next, $index)=@_; (3) Must stor these vars sub { (4) Returns the kernel sub # Code here implements your middleware # %options are lexically accessable here # Execute the next item in the chain $next->(...); (5) Does work and calls the next entry (6) Post work if applicable } } } Top Subroutine The top sub routine (1) can take any arguments you desire and can be called what you like. The idea is it represents your middleware/filter and stores any setup lexically for the maker sub to close over. It returns the maker sub. Maker Subroutine This anonymous sub (2) closes over the variables stored in Top and is the input in to this module (via "register"). When being linked (called) by this modules it is provided two arguments, which is the reference to the next item in the chain and the current middleware index. These MUST be stored to be useful, but can be called anything you like (3). Kernel subroutine This anonymous subroutine (4) actually performs the work of the middleware/filter. After work is done, the next item in the chain is called explictly (5). Any extra work can be performed after the chain is completed after this call (6). LINKING CHAINS Multiple chains of middleware can be linked together. This needs to be done in reverse order. The last segment becomes the $last item when linking the preceding chain and so on. EXAMPLES The synopsis example can be found in the examples directory of this distribution. SEE ALSO Sub::Chain and Sub::Pipeline links together subs. They provide other features that this module does not. These iterate over a list of subroutines, at runtime to achieve named subs etc. This modules pre links subroutines together, reducing over head AUTHOR Ruben Westerberg, REPOSITORTY and BUGS Please report any bugs via git hub: COPYRIGHT AND LICENSE Copyright (C) 2023 by Ruben Westerberg This library is free software; you can redistribute it and/or modify it under the same terms as Perl or the MIT license. DISCLAIMER OF WARRANTIES THIS PACKAGE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.