Fix calculation of PMP of product.
Also add phpunit test to avoid this in future.
This commit is contained in:
parent
73266bb965
commit
4f9aef6150
@ -72,7 +72,7 @@ class MouvementStock extends CommonObject
|
||||
|
||||
// Clean parameters
|
||||
if (empty($price)) $price=0;
|
||||
|
||||
|
||||
if (empty($fk_product)) return 0;
|
||||
|
||||
$now=(! empty($datem) ? $datem : dol_now());
|
||||
@ -87,7 +87,7 @@ class MouvementStock extends CommonObject
|
||||
return -1;
|
||||
}
|
||||
$product->load_stock();
|
||||
|
||||
|
||||
// Define if we must make the stock change (If product type is a service or if stock is used also for services)
|
||||
$movestock=0;
|
||||
if ($product->type != 1 || ! empty($conf->global->STOCK_SUPPORTS_SERVICES)) $movestock=1;
|
||||
@ -101,7 +101,7 @@ class MouvementStock extends CommonObject
|
||||
$origintype = '';
|
||||
$fk_origin = 0;
|
||||
}
|
||||
|
||||
|
||||
$sql = "INSERT INTO ".MAIN_DB_PREFIX."stock_mouvement";
|
||||
$sql.= " (datem, fk_product, fk_entrepot, value, type_mouvement, fk_user_author, label, price, fk_origin, origintype)";
|
||||
$sql.= " VALUES ('".$this->db->idate($now)."', ".$fk_product.", ".$entrepot_id.", ".$qty.", ".$type.",";
|
||||
@ -178,11 +178,11 @@ class MouvementStock extends CommonObject
|
||||
$this->error=$this->db->lasterror();
|
||||
$error = -4;
|
||||
}
|
||||
|
||||
|
||||
$oldqtytouse=($oldqty >= 0?$oldqty:0);
|
||||
// We make a test on oldpmp>0 to avoid to use normal rule on old data with no pmp field defined
|
||||
if ($oldpmp > 0) $newpmp=price2num((($oldqtytouse * $oldpmp) + ($qty * $price)) / ($oldqtytouse + $qty), 'MU');
|
||||
else
|
||||
else
|
||||
{
|
||||
$newpmp=$price; // For this product, PMP was not yet set. We will set it later.
|
||||
}
|
||||
@ -196,7 +196,8 @@ class MouvementStock extends CommonObject
|
||||
}
|
||||
else if ($type == 1 || $type == 2)
|
||||
{
|
||||
// After a stock decrease, we don't change value of PMP for product.
|
||||
// After a stock decrease, we don't change value of PMP for product.
|
||||
$newpmp = $oldpmp;
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -226,7 +227,7 @@ class MouvementStock extends CommonObject
|
||||
{
|
||||
$this->error=$this->db->lasterror();
|
||||
$error = -3;
|
||||
}
|
||||
}
|
||||
else if(empty($fk_product_stock))
|
||||
{
|
||||
$fk_product_stock = $this->db->last_insert_id(MAIN_DB_PREFIX."product_stock");
|
||||
@ -273,9 +274,9 @@ class MouvementStock extends CommonObject
|
||||
|
||||
// Call trigger
|
||||
$result=$this->call_trigger('STOCK_MOVEMENT',$user);
|
||||
if ($result < 0) $error++;
|
||||
if ($result < 0) $error++;
|
||||
// End call triggers
|
||||
|
||||
|
||||
//FIXME: Restore previous value of product_id, entrepot_id, qty if trigger fail
|
||||
}
|
||||
|
||||
@ -415,7 +416,7 @@ class MouvementStock extends CommonObject
|
||||
|
||||
/**
|
||||
* Count number of product in stock before a specific date
|
||||
*
|
||||
*
|
||||
* @param int $productidselected Id of product to count
|
||||
* @param timestamp $datebefore Date limit
|
||||
* @return int Number
|
||||
@ -423,11 +424,11 @@ class MouvementStock extends CommonObject
|
||||
function calculateBalanceForProductBefore($productidselected, $datebefore)
|
||||
{
|
||||
$nb=0;
|
||||
|
||||
|
||||
$sql = 'SELECT SUM(value) as nb from '.MAIN_DB_PREFIX.'stock_mouvement';
|
||||
$sql.= ' WHERE fk_product = '.$productidselected;
|
||||
$sql.= " AND datem < '".$this->db->idate($datebefore)."'";
|
||||
|
||||
|
||||
dol_syslog(get_class($this).__METHOD__.'', LOG_DEBUG);
|
||||
$resql=$this->db->query($sql);
|
||||
if ($resql)
|
||||
@ -436,7 +437,7 @@ class MouvementStock extends CommonObject
|
||||
if ($obj) $nb = $obj->nb;
|
||||
return (empty($nb)?0:$nb);
|
||||
}
|
||||
else
|
||||
else
|
||||
{
|
||||
dol_print_error($this->db);
|
||||
return -1;
|
||||
@ -452,7 +453,7 @@ class MouvementStock extends CommonObject
|
||||
*/
|
||||
function _create_batch($dluo, $qty ) {
|
||||
$pdluo=New Productbatch($this->db);
|
||||
|
||||
|
||||
//Try to find an existing record with batch same batch number or id
|
||||
if (is_numeric($dluo)) {
|
||||
$result=$pdluo->fetch($dluo);
|
||||
@ -495,7 +496,7 @@ class MouvementStock extends CommonObject
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
function get_origin($fk_origin, $origintype) {
|
||||
switch ($origintype) {
|
||||
case 'commande':
|
||||
@ -518,12 +519,12 @@ class MouvementStock extends CommonObject
|
||||
require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
|
||||
$origin = new FactureFournisseur($this->db);
|
||||
break;
|
||||
|
||||
|
||||
default:
|
||||
return '';
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
$origin->fetch($fk_origin);
|
||||
return $origin->getNomUrl(1);
|
||||
}
|
||||
|
||||
@ -164,7 +164,9 @@ class AllTests
|
||||
$suite->addTestSuite('HolidayTest');
|
||||
require_once dirname(__FILE__).'/EntrepotTest.php';
|
||||
$suite->addTestSuite('EntrepotTest');
|
||||
|
||||
require_once dirname(__FILE__).'/MouvementStockTest.php';
|
||||
$suite->addTestSuite('MouvementStockTest');
|
||||
|
||||
require_once dirname(__FILE__).'/CategorieTest.php';
|
||||
$suite->addTestSuite('CategorieTest');
|
||||
|
||||
|
||||
255
test/phpunit/MouvementStockTest.php
Normal file
255
test/phpunit/MouvementStockTest.php
Normal file
@ -0,0 +1,255 @@
|
||||
<?php
|
||||
/* Copyright (C) 2010 Laurent Destailleur <eldy@users.sourceforge.net>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
* or see http://www.gnu.org/
|
||||
*/
|
||||
|
||||
/**
|
||||
* \file test/phpunit/MouvementStockTest.php
|
||||
* \ingroup test
|
||||
* \brief PHPUnit test
|
||||
* \remarks To run this script as CLI: phpunit filename.php
|
||||
*/
|
||||
|
||||
global $conf,$user,$langs,$db;
|
||||
//define('TEST_DB_FORCE_TYPE','mysql'); // This is to force using mysql driver
|
||||
//require_once 'PHPUnit/Autoload.php';
|
||||
require_once dirname(__FILE__).'/../../htdocs/master.inc.php';
|
||||
require_once dirname(__FILE__).'/../../htdocs/product/stock/class/mouvementstock.class.php';
|
||||
require_once dirname(__FILE__).'/../../htdocs/product/stock/class/entrepot.class.php';
|
||||
require_once dirname(__FILE__).'/../../htdocs/product/class/product.class.php';
|
||||
|
||||
if (empty($user->id))
|
||||
{
|
||||
print "Load permissions for admin user nb 1\n";
|
||||
$user->fetch(1);
|
||||
$user->getrights();
|
||||
}
|
||||
$conf->global->MAIN_DISABLE_ALL_MAILS=1;
|
||||
|
||||
|
||||
/**
|
||||
* Class for PHPUnit tests
|
||||
*
|
||||
* @backupGlobals disabled
|
||||
* @backupStaticAttributes enabled
|
||||
* @remarks backupGlobals must be disabled to have db,conf,user and lang not erased.
|
||||
*/
|
||||
class MouvementStockTest extends PHPUnit_Framework_TestCase
|
||||
{
|
||||
protected $savconf;
|
||||
protected $savuser;
|
||||
protected $savlangs;
|
||||
protected $savdb;
|
||||
|
||||
/**
|
||||
* Constructor
|
||||
* We save global variables into local variables
|
||||
*
|
||||
* @return ContratTest
|
||||
*/
|
||||
function __construct()
|
||||
{
|
||||
//$this->sharedFixture
|
||||
global $conf,$user,$langs,$db;
|
||||
$this->savconf=$conf;
|
||||
$this->savuser=$user;
|
||||
$this->savlangs=$langs;
|
||||
$this->savdb=$db;
|
||||
|
||||
print __METHOD__." db->type=".$db->type." user->id=".$user->id;
|
||||
//print " - db ".$db->db;
|
||||
print "\n";
|
||||
}
|
||||
|
||||
// Static methods
|
||||
public static function setUpBeforeClass()
|
||||
{
|
||||
global $conf,$user,$langs,$db;
|
||||
$db->begin(); // This is to have all actions inside a transaction even if test launched without suite.
|
||||
|
||||
print __METHOD__."\n";
|
||||
}
|
||||
|
||||
// tear down after class
|
||||
public static function tearDownAfterClass()
|
||||
{
|
||||
global $conf,$user,$langs,$db;
|
||||
$db->rollback();
|
||||
|
||||
print __METHOD__."\n";
|
||||
}
|
||||
|
||||
/**
|
||||
* Init phpunit tests
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
protected function setUp()
|
||||
{
|
||||
global $conf,$user,$langs,$db;
|
||||
$conf=$this->savconf;
|
||||
$user=$this->savuser;
|
||||
$langs=$this->savlangs;
|
||||
$db=$this->savdb;
|
||||
|
||||
print __METHOD__."\n";
|
||||
}
|
||||
/**
|
||||
* End phpunit tests
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
protected function tearDown()
|
||||
{
|
||||
print __METHOD__."\n";
|
||||
}
|
||||
|
||||
/**
|
||||
* testMouvementCreate
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function testMouvementCreate()
|
||||
{
|
||||
global $conf,$user,$langs,$db;
|
||||
$conf=$this->savconf;
|
||||
$user=$this->savuser;
|
||||
$langs=$this->savlangs;
|
||||
$db=$this->savdb;
|
||||
|
||||
// We create a product for tests
|
||||
$product1=new Product($db);
|
||||
$product1->initAsSpecimen();
|
||||
$product1->ref.=' 1';
|
||||
$product1->label.=' 1';
|
||||
$product1id=$product1->create($user);
|
||||
|
||||
$product2=new Product($db);
|
||||
$product2->initAsSpecimen();
|
||||
$product2->ref.=' 2';
|
||||
$product2->label.=' 2';
|
||||
$product2id=$product2->create($user);
|
||||
|
||||
// We create a product for tests
|
||||
$warehouse1=new Entrepot($db);
|
||||
$warehouse1->initAsSpecimen();
|
||||
$warehouse1->libelle.=' 1';
|
||||
$warehouse1->description.=' 1';
|
||||
$warehouse1id=$warehouse1->create($user);
|
||||
|
||||
$warehouse2=new Entrepot($db);
|
||||
$warehouse2->initAsSpecimen();
|
||||
$warehouse2->libelle.=' 2';
|
||||
$warehouse2->description.=' 2';
|
||||
$warehouse2id=$warehouse2->create($user);
|
||||
|
||||
$localobject=new MouvementStock($this->savdb);
|
||||
//$localobject->initAsSpecimen();
|
||||
|
||||
// Do a list of movement into warehouse 1
|
||||
|
||||
// Create an input movement (type = 3) of price 9.9 -> shoul dupdate PMP to 9.9
|
||||
$result=$localobject->_create($user, $product1id, $warehouse1id, 10, 3, 9.9, 'Movement for unit test 1', 'Inventory Code Test');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
// Create an input movement (type = 3) of price 9.7 -> shoul dupdate PMP to 9.9/9.7 = 9.8
|
||||
$result=$localobject->_create($user, $product1id, $warehouse1id, 10, 3, 9.7, 'Movement for unit test 2', 'Inventory Code Test');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
// Create an output movement (type = 2) of price 9.7 -> shoul dupdate PMP to 9.9/9.7 = 9.8
|
||||
$result=$localobject->_create($user, $product1id, $warehouse1id, -5, 2, 999, 'Movement for unit test 3', 'Inventory Code Test');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
// Create an output movement (type = 1) of price 9.7 -> shoul dupdate PMP to 9.9/9.7 = 9.8
|
||||
$result=$localobject->_create($user, $product1id, $warehouse1id, 1, 0, 0, 'Input from transfer', 'Transfert X');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
// Create an output movement (type = 1) of price 9.7 -> shoul dupdate PMP to 9.9/9.7 = 9.8
|
||||
$result=$localobject->_create($user, $product1id, $warehouse1id, -2, 1, 0, 'Output from transfer', 'Transfert Y');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
|
||||
// Do same but into warehouse 2
|
||||
|
||||
// Create an input movement (type = 3) of price 9.9 -> shoul dupdate PMP to 9.9
|
||||
$result=$localobject->_create($user, $product1id, $warehouse2id, 10, 3, 9.9, 'Movement for unit test 1 wh 2', 'Inventory Code Test 2');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
// Create an input movement (type = 3) of price 9.7 -> shoul dupdate PMP to 9.9/9.7 = 9.8
|
||||
$result=$localobject->_create($user, $product1id, $warehouse2id, 10, 3, 9.7, 'Movement for unit test 2 wh 2', 'Inventory Code Test 2');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
// Create an output movement (type = 2) of price 9.7 -> shoul dupdate PMP to 9.9/9.7 = 9.8
|
||||
$result=$localobject->_create($user, $product1id, $warehouse2id, -5, 2, 999, 'Movement for unit test 3 wh 2', 'Inventory Code Test 2');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
// Create an output movement (type = 1) of price 9.7 -> shoul dupdate PMP to 9.9/9.7 = 9.8
|
||||
$result=$localobject->_create($user, $product1id, $warehouse2id, 1, 0, 0, 'Input from transfer wh 2', 'Transfert X 2');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
// Create an output movement (type = 1) of price 9.7 -> shoul dupdate PMP to 9.9/9.7 = 9.8
|
||||
$result=$localobject->_create($user, $product1id, $warehouse2id, -2, 1, 0, 'Output from transfer wh 2', 'Transfert Y 2');
|
||||
print __METHOD__." result=".$result."\n";
|
||||
$this->assertLessThan($result, 0);
|
||||
|
||||
|
||||
return $localobject;
|
||||
}
|
||||
|
||||
/**
|
||||
* testMouvementCheck
|
||||
*
|
||||
* @param MouvementStock $localobject Movement object we created
|
||||
* @return int
|
||||
*
|
||||
* @depends testMouvementCreate
|
||||
* The depends says test is run only if previous is ok
|
||||
*/
|
||||
public function testMouvementCheck($localobject)
|
||||
{
|
||||
global $conf,$user,$langs,$db;
|
||||
$conf=$this->savconf;
|
||||
$user=$this->savuser;
|
||||
$langs=$this->savlangs;
|
||||
$db=$this->savdb;
|
||||
|
||||
$productid = $localobject->product_id;
|
||||
$warehouseid = $localobject->entrepot_id;
|
||||
|
||||
print __METHOD__." productid=".$productid."\n";
|
||||
|
||||
$producttotest = new Product($db);
|
||||
$producttotest->fetch($productid);
|
||||
|
||||
print __METHOD__." producttotest->stock_reel=".$producttotest->stock_reel."\n";
|
||||
$this->assertEquals($producttotest->stock_reel, 28); // 28 is result of stock movement defined into testMouvementCreate
|
||||
|
||||
print __METHOD__." producttotest->pmp=".$producttotest->pmp."\n";
|
||||
$this->assertEquals($producttotest->pmp, 9.8);
|
||||
|
||||
return $localobject;
|
||||
}
|
||||
|
||||
}
|
||||
Loading…
Reference in New Issue
Block a user