mirror of
https://github.com/Llewellynvdm/php-ml.git
synced 2024-11-21 20:45:10 +00:00
Persistence class to save and restore models (#37)
* Models manager with save/restore capabilities * Refactoring dataset exceptions * Persistency layer docs * New tests for serializable estimators * ModelManager static methods to instance methods
This commit is contained in:
parent
c1b1a5d6ac
commit
8f122fde90
@ -84,6 +84,8 @@ Example scripts are available in a separate repository [php-ai/php-ml-examples](
|
|||||||
* [Iris](machine-learning/datasets/demo/iris/)
|
* [Iris](machine-learning/datasets/demo/iris/)
|
||||||
* [Wine](machine-learning/datasets/demo/wine/)
|
* [Wine](machine-learning/datasets/demo/wine/)
|
||||||
* [Glass](machine-learning/datasets/demo/glass/)
|
* [Glass](machine-learning/datasets/demo/glass/)
|
||||||
|
* Models management
|
||||||
|
* [Persistency](machine-learning/model-manager/persistency/)
|
||||||
* Math
|
* Math
|
||||||
* [Distance](math/distance/)
|
* [Distance](math/distance/)
|
||||||
* [Matrix](math/matrix/)
|
* [Matrix](math/matrix/)
|
||||||
|
24
docs/machine-learning/model-manager/persistency.md
Normal file
24
docs/machine-learning/model-manager/persistency.md
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
# Persistency
|
||||||
|
|
||||||
|
You can save trained models for future use. Persistency across requests achieved by saving and restoring serialized estimators into files.
|
||||||
|
|
||||||
|
### Example
|
||||||
|
|
||||||
|
```
|
||||||
|
use Phpml\Classification\KNearestNeighbors;
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
|
||||||
|
$samples = [[1, 3], [1, 4], [2, 4], [3, 1], [4, 1], [4, 2]];
|
||||||
|
$labels = ['a', 'a', 'a', 'b', 'b', 'b'];
|
||||||
|
|
||||||
|
$classifier = new KNearestNeighbors();
|
||||||
|
$classifier->train($samples, $labels);
|
||||||
|
|
||||||
|
$filepath = '/path/to/store/the/model';
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($classifier, $filepath);
|
||||||
|
|
||||||
|
$restoredClassifier = $modelManager->restoreFromFile($filepath);
|
||||||
|
$restoredClassifier->predict([3, 2]);
|
||||||
|
// return 'b'
|
||||||
|
```
|
@ -40,6 +40,8 @@ pages:
|
|||||||
- Iris: machine-learning/datasets/demo/iris.md
|
- Iris: machine-learning/datasets/demo/iris.md
|
||||||
- Wine: machine-learning/datasets/demo/wine.md
|
- Wine: machine-learning/datasets/demo/wine.md
|
||||||
- Glass: machine-learning/datasets/demo/glass.md
|
- Glass: machine-learning/datasets/demo/glass.md
|
||||||
|
- Models management:
|
||||||
|
- Persistency: machine-learning/model-manager/persistency.md
|
||||||
- Math:
|
- Math:
|
||||||
- Distance: math/distance.md
|
- Distance: math/distance.md
|
||||||
- Matrix: math/matrix.md
|
- Matrix: math/matrix.md
|
||||||
|
@ -4,7 +4,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Phpml\Dataset;
|
namespace Phpml\Dataset;
|
||||||
|
|
||||||
use Phpml\Exception\DatasetException;
|
use Phpml\Exception\FileException;
|
||||||
|
|
||||||
class CsvDataset extends ArrayDataset
|
class CsvDataset extends ArrayDataset
|
||||||
{
|
{
|
||||||
@ -13,16 +13,16 @@ class CsvDataset extends ArrayDataset
|
|||||||
* @param int $features
|
* @param int $features
|
||||||
* @param bool $headingRow
|
* @param bool $headingRow
|
||||||
*
|
*
|
||||||
* @throws DatasetException
|
* @throws FileException
|
||||||
*/
|
*/
|
||||||
public function __construct(string $filepath, int $features, bool $headingRow = true)
|
public function __construct(string $filepath, int $features, bool $headingRow = true)
|
||||||
{
|
{
|
||||||
if (!file_exists($filepath)) {
|
if (!file_exists($filepath)) {
|
||||||
throw DatasetException::missingFile(basename($filepath));
|
throw FileException::missingFile(basename($filepath));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (false === $handle = fopen($filepath, 'rb')) {
|
if (false === $handle = fopen($filepath, 'rb')) {
|
||||||
throw DatasetException::cantOpenFile(basename($filepath));
|
throw FileException::cantOpenFile(basename($filepath));
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($headingRow) {
|
if ($headingRow) {
|
||||||
|
@ -6,15 +6,6 @@ namespace Phpml\Exception;
|
|||||||
|
|
||||||
class DatasetException extends \Exception
|
class DatasetException extends \Exception
|
||||||
{
|
{
|
||||||
/**
|
|
||||||
* @param string $filepath
|
|
||||||
*
|
|
||||||
* @return DatasetException
|
|
||||||
*/
|
|
||||||
public static function missingFile(string $filepath)
|
|
||||||
{
|
|
||||||
return new self(sprintf('Dataset file "%s" missing.', $filepath));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param string $path
|
* @param string $path
|
||||||
@ -25,14 +16,4 @@ class DatasetException extends \Exception
|
|||||||
{
|
{
|
||||||
return new self(sprintf('Dataset root folder "%s" missing.', $path));
|
return new self(sprintf('Dataset root folder "%s" missing.', $path));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $filepath
|
|
||||||
*
|
|
||||||
* @return DatasetException
|
|
||||||
*/
|
|
||||||
public static function cantOpenFile(string $filepath)
|
|
||||||
{
|
|
||||||
return new self(sprintf('Dataset file "%s" can\'t be open.', $filepath));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
39
src/Phpml/Exception/FileException.php
Normal file
39
src/Phpml/Exception/FileException.php
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Phpml\Exception;
|
||||||
|
|
||||||
|
class FileException extends \Exception
|
||||||
|
{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $filepath
|
||||||
|
*
|
||||||
|
* @return FileException
|
||||||
|
*/
|
||||||
|
public static function missingFile(string $filepath)
|
||||||
|
{
|
||||||
|
return new self(sprintf('File "%s" missing.', $filepath));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $filepath
|
||||||
|
*
|
||||||
|
* @return FileException
|
||||||
|
*/
|
||||||
|
public static function cantOpenFile(string $filepath)
|
||||||
|
{
|
||||||
|
return new self(sprintf('File "%s" can\'t be open.', $filepath));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $filepath
|
||||||
|
*
|
||||||
|
* @return FileException
|
||||||
|
*/
|
||||||
|
public static function cantSaveFile(string $filepath)
|
||||||
|
{
|
||||||
|
return new self(sprintf('File "%s" can\'t be saved.', $filepath));
|
||||||
|
}
|
||||||
|
}
|
30
src/Phpml/Exception/SerializeException.php
Normal file
30
src/Phpml/Exception/SerializeException.php
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Phpml\Exception;
|
||||||
|
|
||||||
|
class SerializeException extends \Exception
|
||||||
|
{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $filepath
|
||||||
|
*
|
||||||
|
* @return SerializeException
|
||||||
|
*/
|
||||||
|
public static function cantUnserialize(string $filepath)
|
||||||
|
{
|
||||||
|
return new self(sprintf('"%s" can not be unserialized.', $filepath));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $classname
|
||||||
|
*
|
||||||
|
* @return SerializeException
|
||||||
|
*/
|
||||||
|
public static function cantSerialize(string $classname)
|
||||||
|
{
|
||||||
|
return new self(sprintf('Class "%s" can not be serialized.', $classname));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
52
src/Phpml/ModelManager.php
Normal file
52
src/Phpml/ModelManager.php
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Phpml;
|
||||||
|
|
||||||
|
use Phpml\Estimator;
|
||||||
|
use Phpml\Exception\SerializeException;
|
||||||
|
use Phpml\Exception\FileException;
|
||||||
|
|
||||||
|
class ModelManager
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param Estimator $object
|
||||||
|
* @param string $filepath
|
||||||
|
*/
|
||||||
|
public function saveToFile(Estimator $object, string $filepath)
|
||||||
|
{
|
||||||
|
if (!file_exists($filepath) || !is_writable(dirname($filepath))) {
|
||||||
|
throw FileException::cantSaveFile(basename($filepath));
|
||||||
|
}
|
||||||
|
|
||||||
|
$serialized = serialize($object);
|
||||||
|
if (empty($serialized)) {
|
||||||
|
throw SerializeException::cantSerialize(get_type($object));
|
||||||
|
}
|
||||||
|
|
||||||
|
$result = file_put_contents($filepath, $serialized, LOCK_EX);
|
||||||
|
if ($result === false) {
|
||||||
|
throw FileException::cantSaveFile(basename($filepath));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $filepath
|
||||||
|
*
|
||||||
|
* @return Estimator
|
||||||
|
*/
|
||||||
|
public function restoreFromFile(string $filepath)
|
||||||
|
{
|
||||||
|
if (!file_exists($filepath) || !is_readable($filepath)) {
|
||||||
|
throw FileException::cantOpenFile(basename($filepath));
|
||||||
|
}
|
||||||
|
|
||||||
|
$object = unserialize(file_get_contents($filepath));
|
||||||
|
if ($object === false) {
|
||||||
|
throw SerializeException::cantUnserialize(basename($filepath));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $object;
|
||||||
|
}
|
||||||
|
}
|
@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace tests\Classification;
|
namespace tests\Classification;
|
||||||
|
|
||||||
use Phpml\Association\Apriori;
|
use Phpml\Association\Apriori;
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
|
||||||
class AprioriTest extends \PHPUnit_Framework_TestCase
|
class AprioriTest extends \PHPUnit_Framework_TestCase
|
||||||
{
|
{
|
||||||
@ -184,4 +185,22 @@ class AprioriTest extends \PHPUnit_Framework_TestCase
|
|||||||
|
|
||||||
return $method->invokeArgs($object, $params);
|
return $method->invokeArgs($object, $params);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testSaveAndRestore()
|
||||||
|
{
|
||||||
|
$classifier = new Apriori(0.5, 0.5);
|
||||||
|
$classifier->train($this->sampleGreek, []);
|
||||||
|
|
||||||
|
$testSamples = [['alpha', 'epsilon'], ['beta', 'theta']];
|
||||||
|
$predicted = $classifier->predict($testSamples);
|
||||||
|
|
||||||
|
$filename = 'apriori-test-'.rand(100, 999).'-'.uniqid();
|
||||||
|
$filepath = tempnam(sys_get_temp_dir(), $filename);
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($classifier, $filepath);
|
||||||
|
|
||||||
|
$restoredClassifier = $modelManager->restoreFromFile($filepath);
|
||||||
|
$this->assertEquals($classifier, $restoredClassifier);
|
||||||
|
$this->assertEquals($predicted, $restoredClassifier->predict($testSamples));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace tests\Classification;
|
namespace tests\Classification;
|
||||||
|
|
||||||
use Phpml\Classification\DecisionTree;
|
use Phpml\Classification\DecisionTree;
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
|
||||||
class DecisionTreeTest extends \PHPUnit_Framework_TestCase
|
class DecisionTreeTest extends \PHPUnit_Framework_TestCase
|
||||||
{
|
{
|
||||||
@ -55,6 +56,26 @@ class DecisionTreeTest extends \PHPUnit_Framework_TestCase
|
|||||||
return $classifier;
|
return $classifier;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testSaveAndRestore()
|
||||||
|
{
|
||||||
|
list($data, $targets) = $this->getData($this->data);
|
||||||
|
$classifier = new DecisionTree(5);
|
||||||
|
$classifier->train($data, $targets);
|
||||||
|
|
||||||
|
$testSamples = [['sunny', 78, 72, 'false'], ['overcast', 60, 60, 'false']];
|
||||||
|
$predicted = $classifier->predict($testSamples);
|
||||||
|
|
||||||
|
$filename = 'decision-tree-test-'.rand(100, 999).'-'.uniqid();
|
||||||
|
$filepath = tempnam(sys_get_temp_dir(), $filename);
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($classifier, $filepath);
|
||||||
|
|
||||||
|
$restoredClassifier = $modelManager->restoreFromFile($filepath);
|
||||||
|
$this->assertEquals($classifier, $restoredClassifier);
|
||||||
|
$this->assertEquals($predicted, $restoredClassifier->predict($testSamples));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
public function testTreeDepth()
|
public function testTreeDepth()
|
||||||
{
|
{
|
||||||
list($data, $targets) = $this->getData($this->data);
|
list($data, $targets) = $this->getData($this->data);
|
||||||
|
@ -6,6 +6,7 @@ namespace tests\Classification;
|
|||||||
|
|
||||||
use Phpml\Classification\KNearestNeighbors;
|
use Phpml\Classification\KNearestNeighbors;
|
||||||
use Phpml\Math\Distance\Chebyshev;
|
use Phpml\Math\Distance\Chebyshev;
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
|
||||||
class KNearestNeighborsTest extends \PHPUnit_Framework_TestCase
|
class KNearestNeighborsTest extends \PHPUnit_Framework_TestCase
|
||||||
{
|
{
|
||||||
@ -57,4 +58,27 @@ class KNearestNeighborsTest extends \PHPUnit_Framework_TestCase
|
|||||||
|
|
||||||
$this->assertEquals($testLabels, $predicted);
|
$this->assertEquals($testLabels, $predicted);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testSaveAndRestore()
|
||||||
|
{
|
||||||
|
$trainSamples = [[1, 3], [1, 4], [2, 4], [3, 1], [4, 1], [4, 2]];
|
||||||
|
$trainLabels = ['a', 'a', 'a', 'b', 'b', 'b'];
|
||||||
|
|
||||||
|
$testSamples = [[3, 2], [5, 1], [4, 3], [4, -5], [2, 3], [1, 2], [1, 5], [3, 10]];
|
||||||
|
$testLabels = ['b', 'b', 'b', 'b', 'a', 'a', 'a', 'a'];
|
||||||
|
|
||||||
|
// Using non-default constructor parameters to check that their values are restored.
|
||||||
|
$classifier = new KNearestNeighbors(3, new Chebyshev());
|
||||||
|
$classifier->train($trainSamples, $trainLabels);
|
||||||
|
$predicted = $classifier->predict($testSamples);
|
||||||
|
|
||||||
|
$filename = 'knearest-neighbors-test-'.rand(100, 999).'-'.uniqid();
|
||||||
|
$filepath = tempnam(sys_get_temp_dir(), $filename);
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($classifier, $filepath);
|
||||||
|
|
||||||
|
$restoredClassifier = $modelManager->restoreFromFile($filepath);
|
||||||
|
$this->assertEquals($classifier, $restoredClassifier);
|
||||||
|
$this->assertEquals($predicted, $restoredClassifier->predict($testSamples));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace tests\Classification;
|
namespace tests\Classification;
|
||||||
|
|
||||||
use Phpml\Classification\NaiveBayes;
|
use Phpml\Classification\NaiveBayes;
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
|
||||||
class NaiveBayesTest extends \PHPUnit_Framework_TestCase
|
class NaiveBayesTest extends \PHPUnit_Framework_TestCase
|
||||||
{
|
{
|
||||||
@ -45,4 +46,27 @@ class NaiveBayesTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($testLabels, $classifier->predict($testSamples));
|
$this->assertEquals($testLabels, $classifier->predict($testSamples));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testSaveAndRestore()
|
||||||
|
{
|
||||||
|
$trainSamples = [[5, 1, 1], [1, 5, 1], [1, 1, 5]];
|
||||||
|
$trainLabels = ['a', 'b', 'c'];
|
||||||
|
|
||||||
|
$testSamples = [[3, 1, 1], [5, 1, 1], [4, 3, 8]];
|
||||||
|
$testLabels = ['a', 'a', 'c'];
|
||||||
|
|
||||||
|
$classifier = new NaiveBayes();
|
||||||
|
$classifier->train($trainSamples, $trainLabels);
|
||||||
|
$predicted = $classifier->predict($testSamples);
|
||||||
|
|
||||||
|
$filename = 'naive-bayes-test-'.rand(100, 999).'-'.uniqid();
|
||||||
|
$filepath = tempnam(sys_get_temp_dir(), $filename);
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($classifier, $filepath);
|
||||||
|
|
||||||
|
$restoredClassifier = $modelManager->restoreFromFile($filepath);
|
||||||
|
$this->assertEquals($classifier, $restoredClassifier);
|
||||||
|
$this->assertEquals($predicted, $restoredClassifier->predict($testSamples));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ namespace tests\Classification;
|
|||||||
|
|
||||||
use Phpml\Classification\SVC;
|
use Phpml\Classification\SVC;
|
||||||
use Phpml\SupportVectorMachine\Kernel;
|
use Phpml\SupportVectorMachine\Kernel;
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
|
||||||
class SVCTest extends \PHPUnit_Framework_TestCase
|
class SVCTest extends \PHPUnit_Framework_TestCase
|
||||||
{
|
{
|
||||||
@ -42,4 +43,26 @@ class SVCTest extends \PHPUnit_Framework_TestCase
|
|||||||
|
|
||||||
$this->assertEquals($testLabels, $predictions);
|
$this->assertEquals($testLabels, $predictions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testSaveAndRestore()
|
||||||
|
{
|
||||||
|
$trainSamples = [[1, 3], [1, 4], [2, 4], [3, 1], [4, 1], [4, 2]];
|
||||||
|
$trainLabels = ['a', 'a', 'a', 'b', 'b', 'b'];
|
||||||
|
|
||||||
|
$testSamples = [[3, 2], [5, 1], [4, 3]];
|
||||||
|
$testLabels = ['b', 'b', 'b'];
|
||||||
|
|
||||||
|
$classifier = new SVC(Kernel::LINEAR, $cost = 1000);
|
||||||
|
$classifier->train($trainSamples, $trainLabels);
|
||||||
|
$predicted = $classifier->predict($testSamples);
|
||||||
|
|
||||||
|
$filename = 'svc-test-'.rand(100, 999).'-'.uniqid();
|
||||||
|
$filepath = tempnam(sys_get_temp_dir(), $filename);
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($classifier, $filepath);
|
||||||
|
|
||||||
|
$restoredClassifier = $modelManager->restoreFromFile($filepath);
|
||||||
|
$this->assertEquals($classifier, $restoredClassifier);
|
||||||
|
$this->assertEquals($predicted, $restoredClassifier->predict($testSamples));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,7 +9,7 @@ use Phpml\Dataset\CsvDataset;
|
|||||||
class CsvDatasetTest extends \PHPUnit_Framework_TestCase
|
class CsvDatasetTest extends \PHPUnit_Framework_TestCase
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @expectedException \Phpml\Exception\DatasetException
|
* @expectedException \Phpml\Exception\FileException
|
||||||
*/
|
*/
|
||||||
public function testThrowExceptionOnMissingFile()
|
public function testThrowExceptionOnMissingFile()
|
||||||
{
|
{
|
||||||
|
47
tests/Phpml/ModelManagerTest.php
Normal file
47
tests/Phpml/ModelManagerTest.php
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace tests;
|
||||||
|
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
use Phpml\Regression\LeastSquares;
|
||||||
|
|
||||||
|
class ModelManagerTest extends \PHPUnit_Framework_TestCase
|
||||||
|
{
|
||||||
|
|
||||||
|
public function testSaveAndRestore()
|
||||||
|
{
|
||||||
|
$filename = 'test-save-to-file-'.rand(100, 999).'-'.uniqid();
|
||||||
|
$filepath = tempnam(sys_get_temp_dir(), $filename);
|
||||||
|
|
||||||
|
$obj = new LeastSquares();
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($obj, $filepath);
|
||||||
|
|
||||||
|
$restored = $modelManager->restoreFromFile($filepath);
|
||||||
|
$this->assertEquals($obj, $restored);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @expectedException \Phpml\Exception\FileException
|
||||||
|
*/
|
||||||
|
public function testSaveToWrongFile()
|
||||||
|
{
|
||||||
|
$filepath = sys_get_temp_dir() . DIRECTORY_SEPARATOR . 'unexisting';
|
||||||
|
|
||||||
|
$obj = new LeastSquares();
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($obj, $filepath);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @expectedException \Phpml\Exception\FileException
|
||||||
|
*/
|
||||||
|
public function testRestoreWrongFile()
|
||||||
|
{
|
||||||
|
$filepath = sys_get_temp_dir() . DIRECTORY_SEPARATOR . 'unexisting';
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->restoreFromFile($filepath);
|
||||||
|
}
|
||||||
|
}
|
@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace tests\Regression;
|
namespace tests\Regression;
|
||||||
|
|
||||||
use Phpml\Regression\LeastSquares;
|
use Phpml\Regression\LeastSquares;
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
|
||||||
class LeastSquaresTest extends \PHPUnit_Framework_TestCase
|
class LeastSquaresTest extends \PHPUnit_Framework_TestCase
|
||||||
{
|
{
|
||||||
@ -65,4 +66,28 @@ class LeastSquaresTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals(4094.82, $regression->predict([60000, 1996]), '', $delta);
|
$this->assertEquals(4094.82, $regression->predict([60000, 1996]), '', $delta);
|
||||||
$this->assertEquals(5711.40, $regression->predict([60000, 2000]), '', $delta);
|
$this->assertEquals(5711.40, $regression->predict([60000, 2000]), '', $delta);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testSaveAndRestore()
|
||||||
|
{
|
||||||
|
//https://www.easycalculation.com/analytical/learn-least-square-regression.php
|
||||||
|
$samples = [[60], [61], [62], [63], [65]];
|
||||||
|
$targets = [[3.1], [3.6], [3.8], [4], [4.1]];
|
||||||
|
|
||||||
|
$regression = new LeastSquares();
|
||||||
|
$regression->train($samples, $targets);
|
||||||
|
|
||||||
|
//http://www.stat.wmich.edu/s216/book/node127.html
|
||||||
|
$testSamples = [[9300], [10565], [15000]];
|
||||||
|
$predicted = $regression->predict($testSamples);
|
||||||
|
|
||||||
|
$filename = 'least-squares-test-'.rand(100, 999).'-'.uniqid();
|
||||||
|
$filepath = tempnam(sys_get_temp_dir(), $filename);
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($regression, $filepath);
|
||||||
|
|
||||||
|
$restoredRegression = $modelManager->restoreFromFile($filepath);
|
||||||
|
$this->assertEquals($regression, $restoredRegression);
|
||||||
|
$this->assertEquals($predicted, $restoredRegression->predict($testSamples));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ namespace tests\Regression;
|
|||||||
|
|
||||||
use Phpml\Regression\SVR;
|
use Phpml\Regression\SVR;
|
||||||
use Phpml\SupportVectorMachine\Kernel;
|
use Phpml\SupportVectorMachine\Kernel;
|
||||||
|
use Phpml\ModelManager;
|
||||||
|
|
||||||
class SVRTest extends \PHPUnit_Framework_TestCase
|
class SVRTest extends \PHPUnit_Framework_TestCase
|
||||||
{
|
{
|
||||||
@ -34,4 +35,27 @@ class SVRTest extends \PHPUnit_Framework_TestCase
|
|||||||
|
|
||||||
$this->assertEquals([4109.82, 4112.28], $regression->predict([[60000, 1996], [60000, 2000]]), '', $delta);
|
$this->assertEquals([4109.82, 4112.28], $regression->predict([[60000, 1996], [60000, 2000]]), '', $delta);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testSaveAndRestore()
|
||||||
|
{
|
||||||
|
|
||||||
|
$samples = [[60], [61], [62], [63], [65]];
|
||||||
|
$targets = [3.1, 3.6, 3.8, 4, 4.1];
|
||||||
|
|
||||||
|
$regression = new SVR(Kernel::LINEAR);
|
||||||
|
$regression->train($samples, $targets);
|
||||||
|
|
||||||
|
$testSamples = [64];
|
||||||
|
$predicted = $regression->predict($testSamples);
|
||||||
|
|
||||||
|
$filename = 'svr-test'.rand(100, 999).'-'.uniqid();
|
||||||
|
$filepath = tempnam(sys_get_temp_dir(), $filename);
|
||||||
|
$modelManager = new ModelManager();
|
||||||
|
$modelManager->saveToFile($regression, $filepath);
|
||||||
|
|
||||||
|
$restoredRegression = $modelManager->restoreFromFile($filepath);
|
||||||
|
$this->assertEquals($regression, $restoredRegression);
|
||||||
|
$this->assertEquals($predicted, $restoredRegression->predict($testSamples));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user