| Recommend this page to a friend! | 
|  Download | 
| Info | Example | Screenshots |  Files |  Install with Composer |  Download | Reputation | Support forum | Blog | Links | 
| Last Updated | Ratings | Unique User Downloads | Download Rankings | |||||
| 2025-01-08 (2 months ago)  | Not enough user ratings | Total: 13 | All time:  11,461 This week: 6  | |||||
| Version | License | PHP version | Categories | |||
| multione 1.0.0 | GNU Lesser Genera... | 7 | HTTP, Language, Performance and optim..., P... | 
| Description | Author | |
| This package can execute many tasks in parallel using JavaScript. | 
| 
<?php /** @noinspection PhpUnhandledExceptionInspection */ | 
MultiOne is a library to execute concurrent tasks programmed in PHP using JavaScript.
This library is focused to run local task easily and without much extra boilerplate.
The goal is to be able to run CPU intensive tasks using multiples cores by distributing every task in different workers (what does the job). It also adds a new functionality, it allows to simulate a Command Line Interface (CLI) using the web.

<!-- TOC --> * MultiOne * How to use it? * Logic * How to synchronize and share information between every worker? * Command Line Interface (CLI)
* Example
* Example of multi operations * Changelog <!-- TOC -->
Install this library using Composer
composer require eftec/multione
And call it as follows:
use Eftec\MultiOne\MultiOne;
include __DIR__ . '/../vendor/autoload.php';
$multiOne = MultiOne::Factory(
    1000, // every miliseconds
    basename(__FILE__), // the url to call
    4 // the number of workers
)->setMethod(
    static function($numWorkers):array { // the initial call
        // todo code
        $result=[];
        for($instance=0; $instance<$numWorkers; $instance++) {
            $result[$instance]='initial';
        }
        return $result;
    },
    static function($idWorker, $body):array { // the worker call                
        // some job here
        if($done){
            return ['type'=>'end','result'=>'done','ui'=>'done'];
        }       
        return ['result'=>'working', 'ui'=>"Worker: $idWorker"];
    },
    static function($bodies) { // the worker-end call
        echo "all workers ended";
    }
)->setHtmls(
    '<div id="PREFIX_contentcontent">%s</div>', // how the content is draw
    '<div id="PREFIX_worker_%s"></div><div id="PREFIX_worker_error_%s"></div>', // how every worker is draw
    '<div id="PREFIX_worker_end"></div>', // how the worker-end is draw.
    "loading ?" // the loading ui. If empty, then it don't use loading ui
)->runAuto();
The browser will show the next screen

> In the folder examples you can find examples of code.
There are 3 functions (defined by setMethod) that are called in different stages of the process
setMethod(
    static function($numWorkers):array { 
      // the initial call (returns an array with all the payload for each worker)    
    },
    static function($idWorker, $body):array { 
      // the worker call                       
    },
    static function($bodies) { 
      // the worker-end call, it returns a message.       
    }

You can use the payload to synchronize the operations of every worker.

However, if it is not enough, then you can use the next alternatives:

It is possible to run the library to display a command line interface on web.

In CLI mode, the code calls a single worker only when the user inputs a value in the input text.
MultiOne::Factory(
    1000, // every miliseconds
    basename(__FILE__), // the url to call
)->setMethods(
    static function($numWorkers, $payload): array { // the initial call
        $_SESSION['USERNAME'] = 'admin';
        return ["Start"], // the initial screen
];
    },
    static function($idWorker, $payload): array { // the worker call
        // it reads the data and reduces to 1.
        $username=$_SESSION['USERNAME']??null;
        if($username===null) {
            throw new RuntimeException('no user is not set');
        }
        return MultiOne::msgRun("$username:{$payload['line']}","@$username> ");
    }
)
    ->setCli('> ','100%','600px') // we use a cli
    ->runAuto();
Where:
MultiOne::msgAnswer('run','text to add to the console','a prompt'); // run,full,error
> The text to display could contains colors. Example:
MultiOne::msgAnswer('run','<red>red text</red><byellow>yellow background</byellow>');
You can find this example in the example folder
Lets say we need to execute 100 operations in 4 workers
First, we create the 4 workers
MultiOne::Factory(
    1000, // every miliseconds
    basename(__FILE__), // the url to call
    4     // the number of workers
)
Then, we create the initial load and we split in 4 workers
static function($numWorkers):array { // the initial call
    $result=[];
    $numOper=100;
    $split=ceil($numOper/$numWorkers);
    for($instance=0; $instance<$numWorkers; $instance++) {
        $result[$instance]=['done'=>$split*$instance,'task'=>[
          'init'=>($split*$instance)+$split,
          'end'=>$split*$instance]
          ];
    }
    return $result;
}
Then, every worker must do the job
static function($idWorker, $body):array { // the worker call
    $body['done'] += 1;
    if($body['done']>=$body['task']['end']) {
        $body['done']='ok';
        return MultiOne::msgAnswer('end',$body,'done');
    }
    usleep(random_int(500, 1500));
    MultiOne::msgAnswer('run',$body,"Worker: $idWorker");
}
// And when all jobs are done, we could close it. It could be used to merge all the results. It only needs to return a visual result.
static function($body) { // the worker-end call
    echo "all workers ended";
}
Optionally we could do changes in the UI.
Finally, our code could look as follows:
MultiOne::Factory(
    1000, // every miliseconds
    basename(__FILE__), // the url to call
    4     // the number of workers
)->setMethods(
    static function($numWorkers):array { // the initial call
        $result=[];
        $numOper=100;
        $split=ceil($numOper/$numWorkers);
        for($instance=0; $instance<$numWorkers; $instance++) {
          $result[$instance]=['done'=>$split*$instance,'task'=>[
            'init'=>($split*$instance)+$split,
            'end'=>$split*$instance]
            ];
        }
        return $result;
    },
    static function($idWorker,$body):array { // the worker call
        $body['done'] += 1;
        if($body['done']>=$body['task']['end']) {
            $body['done']='ok';
            return MultiOne::msgAnswer('end',$body,'done');
        }
        usleep(random_int(500, 1500));
        return MultiOne::msgAnswer('run',$body,"#$idWorker {$body['done']}");
    },
    static function($body) { // the worker-end call
        echo "all worker ended";
    }
)->runAuto();
| Screenshots (6) | ||
|  Files (19) | 
| File | Role | Description | ||
|---|---|---|---|---|
|  examples (8 files) | ||||
|  src (1 file, 1 directory) | ||||
|    composer.json | Data | Auxiliary data | ||
|    README.md | Doc. | Documentation | ||
|  Files (19) | / | examples | 
| File | Role | Description | 
|---|---|---|
|    example1.php | Example | Example script | 
|    example2.php | Example | Example script | 
|    example2_error.php | Example | Example script | 
|    example3.php | Example | Example script | 
|    examplecli.php | Example | Example script | 
|    examplecli_simple1.php | Example | Example script | 
|    example_simple1.php | Example | Example script | 
|    example_simple2.php | Example | Example script | 
|  Files (19) | / | src | / | templates | 
| File | Role | Description | 
|---|---|---|
|    template_cli.php | Aux. | Configuration script | 
|    template_multi.php | Aux. | Configuration script | 
| The PHP Classes site has supported package installation using the Composer tool since 2013, as you may verify by reading this instructions page. | 
|  Install with Composer | 
| Version Control | Unique User Downloads | Download Rankings | |||||||||||||||
| 100% | 
 | 
 | 
| Applications that use this package | 
 If you know an application of this package, send a message to the author to add a link here.
 If you know an application of this package, send a message to the author to add a link here.