123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374 |
- <?php
- namespace App\Http\Controllers\Api;
- use App\Model\DispatchSub;
- use App\Model\Equipment;
- use App\Model\SystemL;
- use App\Service\Box\BoxService;
- use App\Service\ProductionOrderService;
- use Illuminate\Support\Facades\Config;
- use Illuminate\Support\Facades\DB;
- class TestController extends BaseController
- {
- public function delTestQ(){
- (new BoxService())->delBoxLock();
- dd(1);
- }
- public function delTestQ2(){
- (new ProductionOrderService())->delBoxLock();
- dd(1);
- }
- public function tt(){dd(1);
- }
- public function test(){dd(1);
- $dispatch = DispatchSub::where('del_time',0)
- ->where('finished_num', '>',0)
- ->where('process_id',14)
- ->select('finished_num','dispatch_time_start as upd_time','device_id')
- ->orderBy('dispatch_time_start','asc')
- ->get()->toArray();
- foreach ($dispatch as $key => $value){
- if(! $value['device_id']) $dispatch[$key]['device_id'] = 12;
- }
- // 生成数据
- $generatedData = (new FeederDataGenerator)->generateData($dispatch);
- dd("ok");
- }
- }
- class FeederDataGenerator
- {
- // 设备配置参数
- const CAPACITY = 0.6; // 容器容量(吨)
- const SUCTION_CAPACITY = 0.1; // 容器容量(吨) 下料
- const SUCTION_WEIGHT = 0.1; // 每次吸料重量(吨)
- const SUCTION_TIME_MIN = 300; // 吸料最短时间(秒) - 5分钟
- const SUCTION_TIME_MAX = 360; // 吸料最长时间(秒) - 6分钟
- const SUCTION_INTERVAL_MIN = 300; // 下次吸料最小间隔(秒) -
- const SUCTION_INTERVAL_MAX = 360; // 下次吸料最大间隔(秒) -
- const DISCHARGE_TIME_MIN = 240; // 下料最短时间(秒) - 4分钟
- const DISCHARGE_TIME_MAX = 300; // 下料最长时间(秒) - 5分钟
- const DISCHARGE_INTERVAL_MIN = 300; // 下次下料最小间隔(秒) -
- const DISCHARGE_INTERVAL_MAX = 360; // 下次下料最大间隔(秒) -
- private $devices = []; // 存储设备数据
- /**
- * 处理原始数据并生成设备运行明细
- * @param array $rawData 原始数据源
- * @return array 生成的设备运行明细数据
- */
- public function generateData(array $rawData): array
- {
- // 首先按设备和日期分组汇总产量
- $this->processRawData($rawData);
- dd($this->devices);
- // 为每个设备生成运行明细
- $result = [];
- foreach ($this->devices as $deviceId => $deviceData) {
- // 为每一天生成数据
- foreach ($deviceData['daily_totals'] as $date => $total) {
- $details = $this->generateDeviceDayData($deviceData['device'], $date, $total);
- $result = array_merge($result, $details);
- }
- }
- try {
- DB::beginTransaction();
- if(! empty($result)){
- $chunkSize = 200; // 每次插入的数量
- foreach (array_chunk($result, $chunkSize) as $chunk) {
- SystemL::insert($chunk); // 使用模型的 insert 方法批量插入
- echo '200条写入成功' . "\n";
- }
- }
- DB::commit();
- }catch (\Throwable $exception){
- DB::rollBack();
- return [false, $exception->getMessage()];
- }
- return [true, ''];
- }
- /**
- * 处理原始数据,按设备和日期汇总产量
- * @param array $rawData 原始数据源
- */
- private function processRawData(array $rawData)
- {
- $map = Equipment::select('id','code','title')->get()->toArray();
- foreach ($map as $value){
- $map1[$value['id']] = $value;
- }
- foreach ($rawData as $item) {
- $deviceId = $item['device_id'];
- $t = $map1[$deviceId];
- $timestamp = $item['upd_time'];
- $finishedNum = floatval($item['finished_num']);
- // 解析日期
- $date = date('Y-m-d', $timestamp);
- // 初始化设备数据
- if (!isset($this->devices[$deviceId])) {
- $this->devices[$deviceId] = [
- 'device' => $t,
- 'total_production' => 0,
- 'daily_totals' => []
- ];
- }
- // 累加总产量和每日产量
- $this->devices[$deviceId]['total_production'] += $finishedNum;
- if (!isset($this->devices[$deviceId]['daily_totals'][$date])) {
- $this->devices[$deviceId]['daily_totals'][$date] = 0;
- }
- $this->devices[$deviceId]['daily_totals'][$date] += $finishedNum;
- }
- }
- /**
- * 为单个设备的一天生成详细的运行数据
- * @param int $deviceId 设备ID
- * @param string $date 日期(格式:YYYY-MM-DD)
- * @param float $totalProduction 当天总产量
- * @return array 生成的详细数据条目
- */
- private function generateDeviceDayData(array $device, string $date, float $totalProduction): array
- {
- $result = [];
- // 计算当天需要完成多少轮完整的吸料-下料循环
- $fullCycles = floor($totalProduction / self::SUCTION_CAPACITY);
- $filledAmount = $fullCycles * self::SUCTION_CAPACITY;
- $remainingWeight = bcsub($totalProduction, $filledAmount,2);
- // 设置当天的起始时间戳(早上7:30)
- $currentTime = strtotime($date . ' 07:30:00');
- // 执行完整循环
- for ($i = 0; $i < $fullCycles; $i++) {
- // 执行一次完整的吸料-下料过程
- $cycleData = $this->generateFullCycle($device, $currentTime);
- $result = array_merge($result, $cycleData);
- // 更新当前时间
- $currentTime = end($cycleData)['time'];
- // // 添加随机间隔时间
- // $interval = rand(self::SUCTION_INTERVAL_MIN, self::SUCTION_INTERVAL_MAX);
- // $currentTime += $interval;
- }
- // 如果有剩余量,执行部分吸料过程
- if ($remainingWeight > 0) {
- $lastKey = array_key_last($result); // 获取最后一个键
- $result[$lastKey]['value'] += $remainingWeight;
- // $remainingCyleData = $this->generatePartialCycle($device, $currentTime, $remainingWeight);
- // $result = array_merge($result, $remainingCyleData);
- }
- return $result;
- }
- /**
- * 随机插入急停事件,并模拟急停持续时间
- *
- * @param int $deviceId 设备ID
- * @param int &$currentTime 当前时间戳(引用传递,用于更新)
- * @param array &$result 结果数组(引用传递,用于添加急停记录)
- */
- private function maybeInsertEmergencyStop(array $device, int &$currentTime, array &$result, $data_point_id, $data_point_name): void
- {
- if (rand(1, 100) <= 3) { // 3% 的概率触发急停
- $emergencyStopTime = $currentTime;
- // 插入急停开始事件
- $result[] = [
- 'time' => $emergencyStopTime,
- 'value' => -1, // 标识为急停
- 'device_no' => $device['code'],
- 'device_name' => $device['title'],
- 'data_point_id' => $data_point_id,
- 'data_point_name' => $data_point_name,
- 'push_time' => $emergencyStopTime,
- // 'push_time_day' => date("Y-m-d H:i:s",$suctionStartTime)
- ];
- // 模拟急停持续时间(10~15秒)
- $stopDuration = rand(10, 15);
- $currentTime += $stopDuration;
- // 可选:插入急停结束事件(也可以不插,只用时间推移表示)
- // $result[] = [
- // 'time' => $currentTime,
- // 'value' => -2, // 急停恢复
- // 'device_no' => $deviceId,
- // 'push_time' => $currentTime
- // ];
- }
- }
- /**
- * 生成一个完整的吸料-下料循环数据
- * @param int $deviceId 设备ID
- * @param int $startTime 起始时间戳
- * @return array 生成的数据条目
- */
- private function generateFullCycle(array $device, int $startTime): array
- {
- $result = [];
- $currentTime = $startTime;
- // 吸料阶段
- for ($i = 0; $i < 1; $i++) {
- // 【吸料前】尝试插入急停
- $this->maybeInsertEmergencyStop($device, $currentTime, $result,5,"吸料急停");
- // 吸料开始
- $suctionStartTime = $currentTime;
- $result[] = [
- 'time' => $suctionStartTime,
- 'value' => 1,
- 'device_no' => $device['code'],
- 'device_name' => $device['title'],
- 'data_point_id' => 1,
- 'data_point_name' => "吸料",
- 'push_time' => $suctionStartTime,
- // 'push_time_day' => date("Y-m-d H:i:s",$suctionStartTime)
- ];
- // 吸料结束
- $suctionDuration = rand(self::SUCTION_TIME_MIN, self::SUCTION_TIME_MAX);
- $currentTime += $suctionDuration;
- // 【吸料后】尝试插入急停
- $this->maybeInsertEmergencyStop($device, $currentTime, $result,5,"吸料急停");
- $result[] = [
- 'time' => $currentTime,
- 'value' => self::SUCTION_WEIGHT,
- 'device_no' => $device['code'],
- 'device_name' => $device['title'],
- 'data_point_id' => 2,
- 'data_point_name' => "吸料结束且本次吸料重量",
- 'push_time' => $currentTime,
- // 'push_time_day' => date("Y-m-d H:i:s",$currentTime)
- ];
- // 添加随机间隔时间
- if ($i < 1) { // 最后一次吸料后不需要等待
- // $interval = rand(self::SUCTION_INTERVAL_MIN, self::SUCTION_INTERVAL_MAX);
- $interval = rand(10, 15);
- $currentTime += $interval;
- }
- }
- // 【下料前】尝试插入急停
- $this->maybeInsertEmergencyStop($device, $currentTime, $result,6,"下料急停");
- // 下料阶段
- $dischargeStartTime = $currentTime;
- $result[] = [
- 'time' => $dischargeStartTime,
- 'value' => 1,
- 'device_no' => $device['code'],
- 'device_name' => $device['title'],
- 'data_point_id' => 3,
- 'data_point_name' => "下料",
- 'push_time' => $dischargeStartTime,
- // 'push_time_day' => date("Y-m-d H:i:s",$dischargeStartTime)
- ];
- $dischargeDuration = rand(self::DISCHARGE_TIME_MIN, self::DISCHARGE_TIME_MAX);
- $currentTime += $dischargeDuration;
- // 【下料后】尝试插入急停
- $this->maybeInsertEmergencyStop($device, $currentTime, $result,6,"下料急停");
- $result[] = [
- 'time' => $currentTime,
- 'value' => self::SUCTION_CAPACITY,
- 'device_no' => $device['code'],
- 'device_name' => $device['title'],
- 'data_point_id' => 4,
- 'data_point_name' => "下料结束且本次下料重量",
- 'push_time' => $currentTime,
- // 'push_time_day' => date("Y-m-d H:i:s",$currentTime)
- ];
- return $result;
- }
- /**
- * 生成一个不完整的循环(只有吸料部分)
- * @param int $deviceId 设备ID
- * @param int $startTime 起始时间戳
- * @param float $requiredWeight 需要吸的重量
- * @return array 生成的数据条目
- */
- private function generatePartialCycle(array $device, int $startTime, float $requiredWeight): array
- {
- $result = [];
- $currentTime = $startTime;
- // 计算需要多少次吸料
- $suctionsNeeded = ceil($requiredWeight / self::SUCTION_WEIGHT);
- $totalSuctionWeight = 0;
- for ($i = 0; $i < $suctionsNeeded; $i++) {
- // 吸料开始
- $suctionStartTime = $currentTime;
- $result[] = [
- 'time' => $suctionStartTime,
- 'value' => 1,
- 'device_no' => $device['code'],
- 'device_name' => $device['title'],
- 'data_point_id' => 1,
- 'data_point_name' => "吸料",
- 'push_time' => $suctionStartTime,
- // 'push_time_day' => => date("Y-m-d H:i:s",$suctionStartTime)
- ];
- // 吸料结束
- $suctionDuration = rand(self::SUCTION_TIME_MIN, self::SUCTION_TIME_MAX);
- $currentTime += $suctionDuration;
- // 判断这次吸料是否会导致超过所需总量
- $addedWeight = ($i == $suctionsNeeded - 1) ? $requiredWeight - $totalSuctionWeight : self::SUCTION_WEIGHT;
- $totalSuctionWeight += $addedWeight;
- $result[] = [
- 'time' => $currentTime,
- 'value' => $addedWeight,
- 'device_no' => $device['code'],
- 'device_name' => $device['title'],
- 'data_point_id' => 2,
- 'data_point_name' => "吸料结束且本次吸料重量",
- 'push_time' => $currentTime,
- // 'push_time_day' => => date("Y-m-d H:i:s",$currentTime)
- ];
- // 添加随机间隔时间(在40-50分钟之间)
- if ($i < $suctionsNeeded - 1) {
- $interval = rand(self::SUCTION_INTERVAL_MIN, self::SUCTION_INTERVAL_MAX);
- $currentTime += $interval;
- }
- }
- return $result;
- }
- }
|