yangys
2025-09-10 dc01577b31204fd3c80645b45d81ff693f79a985
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
 
package org.springblade.mdm.program.service;
 
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.io.FilenameUtils;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils;
import org.springblade.core.log.exception.ServiceException;
import org.springblade.core.mp.base.BizServiceImpl;
import org.springblade.core.oss.OssTemplate;
import org.springblade.core.oss.model.BladeFile;
import org.springblade.core.redis.cache.BladeRedis;
import org.springblade.core.secure.utils.AuthUtil;
import org.springblade.core.tool.utils.DateUtil;
import org.springblade.core.tool.utils.FileUtil;
import org.springblade.core.tool.utils.Func;
import org.springblade.mdm.basesetting.machine.entity.Machine;
import org.springblade.mdm.basesetting.machine.service.MachineService;
import org.springblade.mdm.commons.service.ParamService;
import org.springblade.mdm.flow.entity.FlowProgramFile;
import org.springblade.mdm.flow.service.CureFlowService;
import org.springblade.mdm.flow.service.FlowCommonService;
import org.springblade.mdm.program.entity.DncBackFile;
import org.springblade.mdm.program.entity.NcNode;
import org.springblade.mdm.program.entity.NcProgramExchange;
import org.springblade.mdm.program.mapper.NcProgramExchangeMapper;
import org.springblade.mdm.program.service.programannotation.AnnotationProcessor;
import org.springblade.mdm.program.service.programannotation.DefaultProcessor;
import org.springblade.mdm.program.service.programannotation.FanucProcessor;
import org.springblade.mdm.program.service.programannotation.MachineAnnotationConfig;
import org.springblade.mdm.program.vo.DncSendBackData;
import org.springblade.mdm.program.vo.DncSendBackFile;
import org.springblade.mdm.program.vo.ProgramAnnotation;
import org.springblade.mdm.utils.FileContentUtil;
import org.springblade.mdm.utils.ZipTextFileContentUtil;
import org.springblade.system.pojo.entity.DictBiz;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.multipart.MultipartFile;
 
import java.io.*;
import java.lang.annotation.Annotation;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardOpenOption;
import java.time.Duration;
import java.util.*;
import java.util.regex.Pattern;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
 
/**
 * DNC回传文件处理服务
 *
 * @author yangys
 */
@Slf4j
@Service
@AllArgsConstructor
public class DNCSendBackService extends BizServiceImpl<NcProgramExchangeMapper, NcProgramExchange> {
    private final CureFlowService cureFlowService;
    private final ProgramAnnotationService programAnnotationService;
    private final NcNodeService ncNodeService;
    private final OssTemplate ossTemplate;
    private final BladeRedis bladeRedis;
    private final FlowCommonService flowCommonService;
    private final DncBackFileService dncBackFileService;
    private final MachineService machineService;
    private final ParamService paramService;
    private final MachineAnnotationConfig machineAnnotationConfig;
 
    private final FanucProcessor fanucProcessor;
    private final DefaultProcessor defaultProcessor;
    /**
     * 偏离单文件末尾的模式:P+数字
     */
    private static final String P_NUMBER_PATTERN = "(?i)P\\d+";
 
    private String getFileKey(){
        return "dncimpfile-"+ AuthUtil.getUserId();
    }
    /**
     * dnc回传文件上传
     * @param file DNC回传文件
     * @return 压缩包内程序包名的列表
     */
    public List<DncSendBackData> dncSendBackUpload(MultipartFile file) {
        List<DncSendBackData> list;
        if(file == null || file.isEmpty()){
            throw new ServiceException("文件为空");
        }
 
        if(!StringUtils.endsWith(file.getOriginalFilename(),".zip")){
            throw new ServiceException("文件必须为zip包");
        }
        try {
            BladeFile bfile = ossTemplate.putFile(file);//上传,供后续入库使用
            //设置一个缓存,2小时过期
            bladeRedis.setEx(getFileKey(),bfile.getName(), Duration.ofHours(2));
 
            try(InputStream zipFileInputStream = ossTemplate.statFileStream(bfile.getName());) {
                list = parseProgramListFromZip(zipFileInputStream);
            }
        } catch (IOException e) {
            log.error("上传dnc回传文件失败",e);
            throw new ServiceException("解析DNC回传数据失败");
        }
        return list;
    }
 
 
    /**
     * 从压缩包 解析回传程序列表,这里解析目录即可,目录就是程序包名
     * @param inputStream 压缩包输入流
     * @return 回传程序列表
     * @throws IOException 文件操作异常
     */
    List<DncSendBackData> parseProgramListFromZip(InputStream inputStream) throws IOException {
        List<DncSendBackData> list = new ArrayList<>();
        Path tempZipFile = createTempFile(inputStream);
 
        List<String> fileEntryNameList = new ArrayList<>();
        List<String> dirEntryNameList = new ArrayList<>();//程序包名+工序版次 作为文件夹名
        try (ZipFile zipFile = new ZipFile(tempZipFile.toFile())) {
 
            Enumeration<? extends ZipEntry> zipEntries = zipFile.entries();
            ZipEntry entry;
            //获取所有的entry名称
            while (zipEntries.hasMoreElements()) {
                entry = zipEntries.nextElement();
 
                if(entry.isDirectory()){
                    dirEntryNameList.add(entry.getName());
                }else{
                    fileEntryNameList.add(entry.getName());
                }
            }
 
            List<DictBiz> annotionDictList = programAnnotationService.getAnnotionDictList();
            NcNode programPackageNode;
            //目录列表,即程序包列表
            for(String entryName : dirEntryNameList){
                DncSendBackData progData = new DncSendBackData();
                String folderName = StringUtils.removeEnd(entryName,"/");
 
                //PackageAndProcessEdition pkgAndEdition = parseProgramPackageFromFolderName(folderName);
                String packageName = folderName;//pkgAndEdition.getProgramPackageName();
                //String processEdition = pkgAndEdition.getProcessEdition();
                /*
                if(StringUtils.isBlank(packageName) || StringUtils.isBlank(processEdition)){
                    throw new ServiceException("包内文件夹名格式错误,应该为[零组件号-工序号-工序版次]:"+folderName);
                }*/
 
                progData.setProgramName(packageName);
                String statusLine;
                Optional<String> optFilename = fileEntryNameList.stream().filter(n -> n.startsWith(entryName)).findFirst();
                if(optFilename.isPresent()){
                    entry = zipFile.getEntry(optFilename.get());
                    InputStream ins = zipFile.getInputStream(entry);
                    ByteArrayInputStream bais = new ByteArrayInputStream(IOUtils.toByteArray(ins));;
                    progData.setFileBackTime(DateUtil.fromInstant(entry.getLastModifiedTime().toInstant()));
                    //machineAnnotationConfig.getConfigMap().get(machine.getcon)
                    Pattern fanucPattern = Pattern.compile("^[oO]\\d{4}");
                    String testLine = FileContentUtil.readLineAt(bais,2);//TODO,fanuc这行是下发路径,其他机器是状态
                    bais.reset();
                    //boolean isFanuc = fanucPattern.matcher(testLine).matches();
                    boolean isFanuc = !StringUtils.containsAny(testLine,"(SQ)","(GH)","(PL)");
                    if(isFanuc){
                        statusLine = FileContentUtil.readLineAt(bais,3);//fanuc的状态读第4行
                    }else{
                        statusLine = testLine;
                    }
 
                    if(statusLine.contains("SQ")){
                        //试切
                        programPackageNode = ncNodeService.getLastEditionTryingProgramPackage(packageName);//,processEdition
                    }else if(statusLine.contains("GH")){
                        //固化
                        programPackageNode = ncNodeService.getLastEditionCuredProgramPackage(packageName);
                    }else if(statusLine.contains("PL")){
                        //偏离
                        programPackageNode =ncNodeService.getLastEditionDeviationProgramPackage(packageName);
                    }else{
                        //查询是否车床,是车床可以放过,按试切处理
                        programPackageNode = ncNodeService.getLastEditionTryingProgramPackage(packageName);
                        if(programPackageNode!=null){
                            Machine machine = machineService.getByCode(programPackageNode.getMachineCode());
                            String chechuangVal = paramService.turninngValue();
                            if(StringUtils.equals(machine.getMachineSpec(),chechuangVal)){
                                //车床
                                ProgramAnnotation pa = programAnnotationService.getProgramAnnotationFormat(machine.getControlSystem(),annotionDictList);
                                statusLine = pa.addAnnotation(ProgramAnnotationService.SQ);
                            }
                        }
                    }
 
                    if(programPackageNode != null) {
                        progData.setId(programPackageNode.getId());
                        progData.setProgramNo(programPackageNode.getProgramNo());
 
                        List<String> fileEntryNames  = fileEntryNameList.stream().filter(n -> n.startsWith(packageName)).toList();
                        List<DncSendBackFile> programFiles = new ArrayList<>();
                        fileEntryNames.forEach( filePath ->{
                                DncSendBackFile backFile = new DncSendBackFile();
                                backFile.setEntryName(filePath);
                                backFile.setName(StringUtils.removeStart(filePath,entryName));
                                programFiles.add(backFile);
                        });
 
                        progData.setFiles(programFiles);
                        list.add(progData);
                    }else{
                        throw new ServiceException("找不到程序包名:"+packageName);
                    }
                }else{
                    throw new ServiceException(entryName+"包下未找到文件");
                }
 
            }
        }
 
        return list;
    }
 
    /**
     * 从文件夹名解析出程序包名和和工序版次
     * @param folderName 文件夹名
     * @return 结构数据
     */
    /*
    PackageAndProcessEdition parseProgramPackageFromFolderName(String folderName){
        int index = StringUtils.lastIndexOf(folderName,'-');
        String processEditon = "";
        String packageName = "";
        if(index != -1){
            processEditon = folderName.substring(index+1);
            packageName = folderName.substring(0,index);
        }
 
        PackageAndProcessEdition result = new PackageAndProcessEdition();;
        result.setProgramPackageName(packageName);
        result.setProcessEdition(processEditon);
        return result;
    }*/
 
    /**
     * 入库回传文件,并启动固化流程
     * @param ids id列表逗号分隔,程序包名 节点的id
     */
    @Transactional
    public void dncFileAccept(String ids) throws IOException {
        List<Long> acceptIdList = Func.toLongList(ids);
        //
        NcProgramExchange exchange;
        String filekey = getFileKey();
        String zipFileName = bladeRedis.get(filekey);
        log.info("filekey={},文件名={}",filekey,zipFileName);
 
        Map<Long,List<FlowProgramFile>> pkgIdFileMap = dealWithBackFile(zipFileName,acceptIdList);
 
        cureFlowService.startCureNew(pkgIdFileMap);
 
    }
 
    /**
     * 处理回传文件
     * @param ossFileName
     * @param acceptIdList
     * @return
     * @throws IOException
     */
    private Map<Long, List<FlowProgramFile>> dealWithBackFile(String ossFileName, List<Long> acceptIdList) throws IOException{
        Map<Long, List<FlowProgramFile>> pkgIdFileMap = new HashMap<>();
 
        InputStream inputStream = this.ossTemplate.statFileStream(ossFileName);
        Path tempZipFile = createTempFile(inputStream);
        List<String> entryNameList = new ArrayList<>();
 
        ZipEntry entry;
        try (java.util.zip.ZipFile zipFile = new java.util.zip.ZipFile(tempZipFile.toFile())) {
            Enumeration<? extends ZipEntry> entries = zipFile.entries();
            while(entries.hasMoreElements()) {
                entry = entries.nextElement();
                entryNameList.add(entry.getName());
            }
            log.info("allentrynames:{}",entryNameList);
 
            List<NcNode> allAcceptPackages =  this.ncNodeService.lambdaQuery().in(NcNode::getId,acceptIdList).list();
            //根据内部文件,读取和分析程序包和程序文件数据
            List<String> dirList = entryNameList.stream().filter(s -> s.endsWith("/")).toList();
            for(String dir : dirList){
                String folderName = StringUtils.removeEnd(dir,"/");
 
                //PackageAndProcessEdition pkgAndEdition = folderName;//parseProgramPackageFromFolderName(folderName);
                String programPackageName = folderName; //pkgAndEdition.getProgramPackageName();
 
                Optional<NcNode> optPackageNode = allAcceptPackages.stream().filter(node -> StringUtils.equals(node.getName(),programPackageName)).findFirst();
 
                if(optPackageNode.isEmpty()){
                    throw new ServiceException("找不到程序"+programPackageName);
                }
                NcNode packageNode = optPackageNode.get();
                if(packageNode.hasCured()) {
                    throw new ServiceException(programPackageName + "已经固化,请勿重复入库。");
                }
                //偏离程序判断是否重复回传
                if(packageNode.isDeviationProgram() && packageNode.hasLocked()) {
                    throw new ServiceException(programPackageName + "已锁定的程序不可以再次回传。");
                }
                //检查是否在审批过程中
                boolean active = flowCommonService.isProcessInstanceActive(packageNode.getProcessInstanceId());
                if(active){
                    throw new ServiceException(programPackageName+"正在审批中,请勿等待审批完成。");
                }
 
                //验证都过了,保存dncbackFile
                DncBackFile backFile = new DncBackFile();
                backFile.setNcNodeId(packageNode.getId());
                backFile.setOssName(ossFileName);
                dncBackFileService.save(backFile);
 
                List<FlowProgramFile> flowFiles = new ArrayList<>();
                //查找包下的文件数据,
                entryNameList.stream().filter(s -> s.startsWith(dir)).forEach(entryName -> {
                    log.info("{}下的文件:{}",dir,entryName);
 
                    if(!entryName.endsWith("/")){
                        //实际的文件
                        String fileName = StringUtils.removeStart(entryName,dir);//去除文件名路径部分
                        fileName = removeDeviationPart(fileName);
                        try {
                            FlowProgramFile newFlowFile = new FlowProgramFile();
                            newFlowFile.setProgramName(packageNode.getName());
                            newFlowFile.setProcessInstanceId(null);//先置为空,启动流程后设置该值
                            newFlowFile.setFileType("program");
                            newFlowFile.setName(fileName);
                            InputStream ins = zipFile.getInputStream(zipFile.getEntry(entryName));
                            BladeFile newOssFile = ossTemplate.putFile("mdm",fileName,ins);
                            newFlowFile.setOssName(newOssFile.getName());
 
                            flowFiles.add(newFlowFile);
                        } catch (IOException e) {
                            throw new RuntimeException(e);
                        }
                    }
 
                });
 
                pkgIdFileMap.put(packageNode.getId(),flowFiles);
            }
        }
 
        return pkgIdFileMap;
    }
 
    String removeDeviationPart(String filename){
        String finalFilename  = filename;
        //去掉文件名中可能带有的偏离单部分:-P[序号]
        String ext = FilenameUtils.getExtension(filename);
        String dotExt = StringUtils.isNotBlank(ext)?"."+ext:ext;//带点的扩展名
        String notExtName = StringUtils.removeEnd(filename,dotExt);
        int idx = notExtName.lastIndexOf("-");
        if(idx != -1){
            String endPart = notExtName.substring(idx+1);
            //Pattern.CASE_INSENSITIVE
            boolean containsPld = endPart.matches(P_NUMBER_PATTERN);
            if(containsPld){
                finalFilename = notExtName.substring(0, idx)+dotExt;
            }
        }
 
        return finalFilename;
    }
    /**
     * 创建一个临时zip文件
     * @param inputStream 文件的输入流
     * @return path 文件
     * @throws IOException
     */
    Path createTempFile(InputStream inputStream) throws IOException {
        byte[] zipData = FileUtil.copyToByteArray(inputStream);
        Path tempFile = Files.createTempFile("tempzip"+System.currentTimeMillis(), ".zip");
        // 写入字节数据到临时文件
        Files.write(tempFile, zipData, StandardOpenOption.WRITE);
 
        return tempFile;
    }
 
    /**
     * 获取回传文件的内容
     * @param entryName 文件在压缩包内的路径
     * @return 文件内容文本
     */
    public String getEntryFileContent(String entryName) throws IOException {
        String result  = "";
        String zipFileName = bladeRedis.get(getFileKey());
        return ZipTextFileContentUtil.getTextContent(this.ossTemplate.statFileStream(zipFileName),entryName);
 
        /*
        try(InputStream inputStream = this.ossTemplate.statFileStream(zipFileName);){
            Path tempZipFile = createTempFile(inputStream);
 
            ZipEntry entry;
            try (java.util.zip.ZipFile zipFile = new java.util.zip.ZipFile(tempZipFile.toFile())) {
                Enumeration<? extends ZipEntry> entries = zipFile.entries();
                while (entries.hasMoreElements()) {
                    entry = entries.nextElement();
                    if (!entryName.equals(entry.getName())) {
                        continue;
                    }
                    try (InputStream fileIns = zipFile.getInputStream(zipFile.getEntry(entryName))) {
                        ByteArrayInputStream bos = new ByteArrayInputStream(fileIns.readAllBytes());
                        boolean isText = StringUtils.endsWithIgnoreCase(entryName,".txt") || StringUtils.endsWithIgnoreCase(entryName,".nc")|| StringUtils.endsWithIgnoreCase(entryName,".xml");
                        if(!isText) {
                            isText = FileContentUtil.isTextFile(bos);
                        }
                        if (isText) {
                            bos.reset();
                            result = FileContentUtil.getContentFromStream(bos);
                        } else {
                            result = "<非文本文件>";
                        }
                    }
 
                }
            }
        }
 
        return result;*/
    }
 
}
 
@Data
class PackageAndProcessEdition{
    private String programPackageName;
    private String processEdition;
 
    public String programName(){
        return programPackageName+"-"+processEdition;
    }
}