289 lines
14 KiB
PHP

<?php
namespace app\controller;
use app\controller\Acl;
use think\facade\Db;
class Summary extends Acl {
public function __construct(){}
//初始化
public function init() {
$period=getPeriod();
//查询条数
$count=Db::name('room_info')->where([['time','>',$period]])->count();
//计价方式
$fun=getSys('fun');
$info=[
'valuation'=>['base'=>'基础计价','ma'=>'移动平均','fifo'=>'先进先出'][$fun['valuation']],
'branch'=>['总仓核算','分仓核算'][$fun['branch']],
'rule'=>['def'=>'结存结余','attr'=>'辅助属性','batch'=>'批次日期','aab'=>'属性批次'][$fun['rule']],
];
//初始结账
$summary=Db::name('summary')->where([['time','>',$period]])->field(['id'])->select()->toArray();
$fifo=Db::name('fifo')->where([['out','in',array_column($summary,'id')]])->select()->toArray();
$relation=[];
foreach ($fifo as $v) {
$relation[]=['id'=>$v['in'],'handle'=>$v['handle']];
}
if(!empty($relation)){
Db::name('summary')->duplicate(['handle'=>Db::raw('handle - VALUES(`handle`)')])->insertAll($relation);
Db::name('fifo')->where([['id','in',array_column($fifo,'id')]])->delete();
}
Db::name('summary')->where([['id','in',array_column($summary,'id')]])->delete();
pushLog('执行数据校准');//日志
//返回数据
$result=[
'state'=>'success',
'count'=>$count,
'info'=>$info
];
return json($result);
}
//收发处理
public function note($type,$class,$mold){
//场景判断
if($mold){
//记录
$info=Db::name('room_info')->where([['type','=',$type],['class','=',$class]])->order(['time','id'])->field(['id'])->select()->toArray();
$this->handle(array_column($info,'id'));
}else{
//清除
$summary=Db::name('summary')->where([['type','=',$type],['class','=',$class]])->field(['id'])->select()->toArray();
$fifo=Db::name('fifo')->where([['out','in',array_column($summary,'id')]])->select()->toArray();
$relation=[];
foreach ($fifo as $v) {
$relation[]=['id'=>$v['in'],'handle'=>$v['handle']];
}
if(!empty($relation)){
Db::name('summary')->duplicate(['handle'=>Db::raw('handle - VALUES(`handle`)')])->insertAll($relation);
Db::name('fifo')->where([['id','in',array_column($fifo,'id')]])->delete();
}
Db::name('summary')->where([['id','in',array_column($summary,'id')]])->delete();
}
}
//轮询数据
public function poll() {
$input=input('post.');
if(existFull($input,['page','limit'])){
Db::startTrans();
try {
$period=getPeriod();
$info=Db::name('room_info')->where([['time','>',$period]])->page($input['page'],$input['limit'])->order(['time','id'])->field(['id'])->select()->toArray();
$this->handle(array_column($info,'id'));
Db::commit();
$result=['state'=>'success'];
} catch (\Exception $e) {
dd($e);
Db::rollback();
$result=['state'=>'error','info'=>'内部错误,操作已撤销!'];
}
}else{
$result=['state'=>'error','info'=>'传入参数不完整!'];
}
return json($result);
}
//收发记录
public function handle($arr){
if(empty($arr)) return;
//查询记录
$info = Db::name('room_info')->where([['id','in',$arr]])->order(['time','id'])->select()->toArray();
//匹配单据
$union=[];
$tab=['buy'=>'buy','bre'=>'bre','sell'=>'sell','sre'=>'sre','swapOut'=>'swap','swapEnter'=>'swap','entry'=>'entry','extry'=>'extry'];
foreach ($tab as $t=>$m) {
$gather=search($info)->where([['type','=',$t]])->select();
if(!empty($gather)){
$union[]=Db::name($m.'_info')->where([['id','in',array_column($gather,'info')]])->fieldRaw('"'.$t.'" as mold,id,goods,attr,'.($t=='swapEnter'?'storehouse as warehouse':'warehouse').',batch,mfd,serial')->buildSql();
}
}
//合并子查询
$union=implode(' UNION ALL ',$union);
$record=DB::query('SELECT * FROM ('.$union.') as nodcloud');
//构造数据
$summary=[];
//exist结存|balance结余
//[0,0,0,0]=[商品总仓|商品分仓|规则总仓|规则分仓]
$def=['exist'=>[0,0,0,0],'balance'=>[0,0,0,0]];
foreach ($info as $vo) {
$row=search($record)->where([['mold','=',$vo['type']],['id','=',$vo['info']]])->find();
$summary[]=[
'pid'=>$vo['id'],
'type'=>$vo['type'],
'class'=>$vo['class'],
'info'=>$vo['info'],
'time'=>$vo['time'],
'goods'=>$row['goods'],
'attr'=>$row['attr'],
'warehouse'=>$row['warehouse'],
'batch'=>$row['batch'],
'mfd'=>$row['mfd'],
'serial'=>$row['serial'],
'direction'=>$vo['direction'],
'price'=>$vo['price'],
'nums'=>$vo['nums'],
'uct'=>0,
'bct'=>0,
'exist'=>json_encode($def['exist']),
'balance'=>json_encode($def['balance']),
'handle'=>0
];
}
Db::name('summary')->insertAll($summary);
$summary=Db::name('summary')->where([['pid','in',array_column($summary,'pid')]])->order(['id'])->select()->toArray();
//处理数据
$fun=getSys('fun');
$goods=Db::name('goods')->where([['id','in',array_column($summary,'goods')]])->field(['id','buy'])->select()->toArray();
$attr=Db::name('attr')->where([['pid','in',array_column($summary,'goods')]])->field(['pid','name','buy'])->select()->toArray();
foreach ($summary as $vo) {
$sql=[
[['id','<',$vo['id']],['goods','=',$vo['goods']]],
[['warehouse','=',$vo['warehouse']]]
];
//规则语句
if($fun['rule']=='def'){
$sql[]=[];
}else if($fun['rule']=='attr'){
$sql[]=[['attr','=',$vo['attr']]];
}else if($fun['rule']=='batch'){
$sql[]=[['batch','=',$vo['batch']],['mfd','=',$vo['mfd']]];
}else{
$sql[]=[['attr','=',$vo['attr']],['batch','=',$vo['batch']],['mfd','=',$vo['mfd']]];
}
//[商品总仓|商品分仓|规则总仓|规则分仓]
$senten=[$sql[0],array_merge($sql[0],$sql[1]),array_merge($sql[0],$sql[2]),array_merge($sql[0],$sql[1],$sql[2])];
$first=[];
$first[]=Db::name('summary')->where($senten[0])->order(['id'=>'DESC'])->find();
$first[]=Db::name('summary')->where($senten[1])->order(['id'=>'DESC'])->find();
$first[]=Db::name('summary')->where($senten[2])->order(['id'=>'DESC'])->find();
$first[]=Db::name('summary')->where($senten[3])->order(['id'=>'DESC'])->find();
//默认值
foreach ($first as $k=>$v) {
if(empty($v)){
$first[$k]=$def;
}else{
$first[$k]=['exist'=>json_decode($v['exist']),'balance'=>json_decode($v['balance'])];
}
}
//数据处理
$g=search($goods)->where([['id','=',$vo['goods']]])->find();
$a=search($attr)->where([['pid','=',$vo['goods']],['name','=',$vo['attr']]])->find();
$buy=empty($a)?$g['buy']:$a['buy'];
//序列判断
$serial=json_decode($vo['serial']);
if(empty($serial)){
//计价方法
if($fun['valuation']=='base'){
//基础计价法
if(in_array($vo['type'],['buy','bre','swapOut','swapEnter','entry','extry'])){
$uct=$vo['price'];
}else{
$uct=$buy;
}
}else if($fun['valuation']=='ma'){
//移动平均法
if(in_array($vo['type'],['buy','bre','swapOut','swapEnter','entry','extry'])){
$uct=$vo['price'];
}else{
//[空|负]库存取采购价
//正常库存取结余除结存
if(empty($fun['branch'])){
$uct=$first[2]['exist'][2]<=0?$buy:math()->chain($first[2]['balance'][2])->div($first[2]['exist'][2])->done();
}else{
$uct=$first[3]['exist'][3]<=0?$buy:math()->chain($first[3]['balance'][3])->div($first[3]['exist'][3])->round(2)->done();
}
}
}else{
//先进先出法
if(in_array($vo['type'],['buy','swapEnter','entry'])){
$uct=$vo['price'];
}else if(in_array($vo['type'],['sre','vre'])){
$uct=$buy;
}else{
$where=[
['id','<',$vo['id']],
['goods','=',$vo['goods']],
['attr','=',$vo['attr']],
['batch','=',$vo['batch']],
['mfd','=',$vo['mfd']],
['direction','=',1],
[DB::raw('nums'),'<>',DB::raw('handle')]
];
empty($fun['branch'])&&$where[]=['warehouse','=',$vo['warehouse']];
$build=DB::name('summary')->where($where)->fieldRaw('id,uct,(nums - handle) as usable,(@sum := @sum + (nums - handle)) as sum')->order('id')->buildSql(false);
$build=str_replace("WHERE","CROSS JOIN ( SELECT @sum := 0 ) t WHERE",$build);
$list=DB::query('SELECT * FROM ('.$build.') as nodcloud WHERE sum < '.$vo['nums'].' or (sum >= '.$vo['nums'].' and sum - usable < '.$vo['nums'].');');
if(empty($list)){
//[无入库]取采购价
$uct=$buy;
}else{
$uct=0;
$knot=$vo['nums'];
$relation=[];
foreach ($list as $v) {
if($knot<=$v['usable']){
$relation[]=['id'=>$v['id'],'handle'=>$knot];
$calc=math()->chain($knot)->mul($v['uct'])->done();
$uct=math()->chain($uct)->add($calc)->done();
break;
}else{
$relation[]=['id'=>$v['id'],'handle'=>$v['usable']];
$calc=math()->chain($v['usable'])->mul($v['uct'])->done();
$uct=math()->chain($uct)->add($calc)->done();
$knot=math()->chain($knot)->sub($v['usable'])->done();
}
}
$uct=math()->chain($uct)->div($vo['nums'])->done();
Db::name('summary')->duplicate(['handle'=>Db::raw('handle + VALUES(`handle`)')])->insertAll($relation);
$fifo=[];
foreach ($relation as $v) {
$fifo[]=['out'=>$vo['id'],'in'=>$v['id'],'handle'=>$v['handle']];
}
Db::name('fifo')->insertAll($fifo);
}
}
}
}else{
//序列产品
if(in_array($vo['type'],['buy','swapEnter','entry'])){
//[无入库]取采购价
$uct=$vo['price'];
}else{
$uct=0;
foreach ($serial as $v) {
$row=DB::name('summary')->where([
['id','<',$vo['id']],
['goods','=',$vo['goods']],
['serial','like','%"'.$v.'"%']
])->field(['uct'])->order(['id'=>'DESC'])->find();
$uct=math()->chain($uct)->add(empty($row)?$buy:$row['uct'])->done();
}
$uct=math()->chain($uct)->div($vo['nums'])->done();
}
}
//综合处理
$uct=math()->chain($uct)->round($fun['digit']['money'])->done();
$exist=[
empty($vo['direction'])?math()->chain($first[0]['exist'][0])->sub($vo['nums'])->done():math()->chain($first[0]['exist'][0])->add($vo['nums'])->done(),
empty($vo['direction'])?math()->chain($first[1]['exist'][1])->sub($vo['nums'])->done():math()->chain($first[1]['exist'][1])->add($vo['nums'])->done(),
empty($vo['direction'])?math()->chain($first[2]['exist'][2])->sub($vo['nums'])->done():math()->chain($first[2]['exist'][2])->add($vo['nums'])->done(),
empty($vo['direction'])?math()->chain($first[3]['exist'][3])->sub($vo['nums'])->done():math()->chain($first[3]['exist'][3])->add($vo['nums'])->done()
];
$bct=math()->chain($uct)->mul($vo['nums'])->done();
$balance=[
empty($vo['direction'])?math()->chain($first[0]['balance'][0])->sub($bct)->done():math()->chain($first[0]['balance'][0])->add($bct)->done(),
empty($vo['direction'])?math()->chain($first[1]['balance'][1])->sub($bct)->done():math()->chain($first[1]['balance'][1])->add($bct)->done(),
empty($vo['direction'])?math()->chain($first[2]['balance'][2])->sub($bct)->done():math()->chain($first[2]['balance'][2])->add($bct)->done(),
empty($vo['direction'])?math()->chain($first[3]['balance'][3])->sub($bct)->done():math()->chain($first[3]['balance'][3])->add($bct)->done()
];
foreach ($exist as $k=>$v){
$v==0&&$balance[$k]=0;
}
$exist=json_encode($exist);
$balance=json_encode($balance);
Db::name('summary')->where([['id','=',$vo['id']]])->update(['uct'=>$uct,'bct'=>$bct,'exist'=>$exist,'balance'=>$balance]);
}
}
}