- Updated Detail class to include JsonSourceGenerationOptions and JsonSerializable attributes for source generation. - Modified ProcessData class to utilize source generation context for Description deserialization. - Renamed RowSourceGenerationContext to QS408MRowSourceGenerationContext for clarity. - Updated Run class to use QS408MRunSourceGenerationContext for serialization. - Enhanced Description class with source generation attributes. - Refactored FileRead class to use source generation context for Description deserialization. - Added new methods in ProcessDataStandardFormat for JSON element array extraction. - Introduced new PDSF file handler classes: Body, Constant, FileRead, Footer, Header, Row, Run, and Site. - Implemented logic for parsing and handling PDSF data structures. - Added unit tests for PDSF processing to ensure functionality.
		
			
				
	
	
		
			230 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			230 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Adaptation.Eaf.Management.ConfigurationData.CellAutomation;
 | |
| using Adaptation.Ifx.Eaf.EquipmentConnector.File.Configuration;
 | |
| using Adaptation.Shared;
 | |
| using Adaptation.Shared.Duplicator;
 | |
| using Adaptation.Shared.Methods;
 | |
| using Adaptation.Shared.Metrology;
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Globalization;
 | |
| using System.IO;
 | |
| using System.Linq;
 | |
| using System.Text;
 | |
| using System.Text.Json;
 | |
| 
 | |
| namespace Adaptation.FileHandlers.OpenInsight;
 | |
| 
 | |
| public class FileRead : Shared.FileRead, IFileRead
 | |
| {
 | |
| 
 | |
|     private readonly string _IqsConnectionString;
 | |
|     private readonly string _OpenInsightFilePattern;
 | |
|     private readonly string _OpenInsightApiECDirectory;
 | |
| 
 | |
|     public FileRead(ISMTP smtp, Dictionary<string, string> fileParameter, string cellInstanceName, int? connectionCount, string cellInstanceConnectionName, FileConnectorConfiguration fileConnectorConfiguration, string equipmentTypeName, string parameterizedModelObjectDefinitionType, IList<ModelObjectParameterDefinition> modelObjectParameters, string equipmentDictionaryName, Dictionary<string, List<long>> dummyRuns, Dictionary<long, List<WS.Results>> staticRuns, bool useCyclicalForDescription, bool isEAFHosted) :
 | |
|         base(new Description(), false, smtp, fileParameter, cellInstanceName, connectionCount, cellInstanceConnectionName, fileConnectorConfiguration, equipmentTypeName, parameterizedModelObjectDefinitionType, modelObjectParameters, equipmentDictionaryName, dummyRuns, staticRuns, useCyclicalForDescription, isEAFHosted: connectionCount is null)
 | |
|     {
 | |
|         _MinFileLength = 10;
 | |
|         _Logistics = new(this);
 | |
|         _NullData = string.Empty;
 | |
|         if (_FileParameter is null)
 | |
|             throw new Exception(cellInstanceConnectionName);
 | |
|         if (_ModelObjectParameterDefinitions is null)
 | |
|             throw new Exception(cellInstanceConnectionName);
 | |
|         if (!_IsDuplicator)
 | |
|             throw new Exception(cellInstanceConnectionName);
 | |
|         _IqsConnectionString = GetPropertyValue(cellInstanceConnectionName, modelObjectParameters, "IQS.ConnectionString");
 | |
|         _OpenInsightFilePattern = GetPropertyValue(cellInstanceConnectionName, modelObjectParameters, "OpenInsight.FilePattern");
 | |
|         _OpenInsightApiECDirectory = GetPropertyValue(cellInstanceConnectionName, modelObjectParameters, "OpenInsight.Api.EC.Directory");
 | |
|     }
 | |
| 
 | |
|     void IFileRead.Move(Tuple<string, Test[], JsonElement[], List<FileInfo>> extractResults, Exception exception)
 | |
|     {
 | |
|         bool isErrorFile = exception is not null;
 | |
|         if (!isErrorFile && !string.IsNullOrEmpty(_Logistics.ReportFullPath))
 | |
|         {
 | |
|             FileInfo fileInfo = new(_Logistics.ReportFullPath);
 | |
|             if (fileInfo.Exists && fileInfo.LastWriteTime < fileInfo.CreationTime)
 | |
|                 File.SetLastWriteTime(_Logistics.ReportFullPath, fileInfo.CreationTime);
 | |
|         }
 | |
|         Move(extractResults);
 | |
|     }
 | |
| 
 | |
|     void IFileRead.WaitForThread() => WaitForThread(thread: null, threadExceptions: null);
 | |
| 
 | |
|     string IFileRead.GetEventDescription()
 | |
|     {
 | |
|         string result = _Description.GetEventDescription();
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     List<string> IFileRead.GetHeaderNames()
 | |
|     {
 | |
|         List<string> results = _Description.GetHeaderNames();
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|     string[] IFileRead.Move(Tuple<string, Test[], JsonElement[], List<FileInfo>> extractResults, string to, string from, string resolvedFileLocation, Exception exception)
 | |
|     {
 | |
|         string[] results = Move(extractResults, to, from, resolvedFileLocation, exception);
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|     JsonProperty[] IFileRead.GetDefault()
 | |
|     {
 | |
|         JsonProperty[] results = _Description.GetDefault(this, _Logistics);
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|     Dictionary<string, string> IFileRead.GetDisplayNamesJsonElement()
 | |
|     {
 | |
|         Dictionary<string, string> results = _Description.GetDisplayNamesJsonElement(this);
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|     List<IDescription> IFileRead.GetDescriptions(IFileRead fileRead, List<Test> tests, IProcessData processData)
 | |
|     {
 | |
|         List<IDescription> results = _Description.GetDescriptions(fileRead, _Logistics, tests, processData);
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|     Tuple<string, Test[], JsonElement[], List<FileInfo>> IFileRead.GetExtractResult(string reportFullPath, string eventName)
 | |
|     {
 | |
|         Tuple<string, Test[], JsonElement[], List<FileInfo>> results;
 | |
|         if (string.IsNullOrEmpty(eventName))
 | |
|             throw new Exception();
 | |
|         _ReportFullPath = reportFullPath;
 | |
|         DateTime dateTime = DateTime.Now;
 | |
|         results = GetExtractResult(reportFullPath, dateTime);
 | |
|         if (results.Item3 is null)
 | |
|             results = new Tuple<string, Test[], JsonElement[], List<FileInfo>>(results.Item1, Array.Empty<Test>(), JsonSerializer.Deserialize<JsonElement[]>("[]"), results.Item4);
 | |
|         if (results.Item3.Length > 0 && _IsEAFHosted)
 | |
|             WritePDSF(this, results.Item3);
 | |
|         UpdateLastTicksDuration(DateTime.Now.Ticks - dateTime.Ticks);
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|     Tuple<string, Test[], JsonElement[], List<FileInfo>> IFileRead.ReExtract()
 | |
|     {
 | |
|         Tuple<string, Test[], JsonElement[], List<FileInfo>> results;
 | |
|         List<string> headerNames = _Description.GetHeaderNames();
 | |
|         Dictionary<string, string> keyValuePairs = _Description.GetDisplayNamesJsonElement(this);
 | |
|         results = ReExtract(this, headerNames, keyValuePairs);
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|     private static string GetLines(Logistics logistics, List<QS408M.Description> descriptions, bool isStratusQual)
 | |
|     {
 | |
|         StringBuilder results = new();
 | |
|         char del = '\t';
 | |
|         QS408M.Description x = descriptions[0];
 | |
|         if (isStratusQual)
 | |
|         {
 | |
|             _ = results.Append("Stratus_").Append(logistics.MID).Append('_').Append(logistics.DateTimeFromSequence.ToString("yyyyMMddhhmmssfff")).Append(del).
 | |
|                 Append(x.Date).Append(del).
 | |
|                 Append(logistics.JobID).Append(del).
 | |
|                 Append("FQA Thickness").Append(del).
 | |
|                 Append(x.Employee).Append(del).
 | |
|                 Append(x.Recipe).Append(del).
 | |
|                 Append(x.Reactor).Append(del).
 | |
|                 Append(x.RDS).Append(del).
 | |
|                 Append(x.PSN).Append(del).
 | |
|                 Append(x.Lot).Append(del).
 | |
|                 Append(x.Cassette).Append(del).
 | |
|                 Append(x.MeanThickness);
 | |
|             for (int i = 0; i < descriptions.Count; i++)
 | |
|                 _ = results.Append(del).Append(descriptions[i].Slot).
 | |
|                             Append(del).Append(descriptions[i].Thickness);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             _ = results.Append("Bio-Rad ").Append(x.UniqueId).Append(del).
 | |
|                 Append(x.Date).Append(del).
 | |
|                 Append(x.ThicknessFourteenCriticalPointsAverage).Append(del).
 | |
|                 Append(x.Recipe).Append(del).
 | |
|                 Append(x.Reactor).Append(del).
 | |
|                 Append(x.RDS).Append(del).
 | |
|                 Append(x.PSN).Append(del).
 | |
|                 Append(x.Layer).Append(del).
 | |
|                 Append(x.Zone).Append(del).
 | |
|                 Append(x.Cassette).Append(del).
 | |
|                 Append(x.Wafer).Append(del).
 | |
|                 Append(x.RVThickness);
 | |
|             for (int i = 0; i < descriptions.Count; i++)
 | |
|                 _ = results.Append(del).Append(descriptions[i].Position).
 | |
|                             Append(del).Append(descriptions[i].Thickness);
 | |
|         }
 | |
|         return results.ToString();
 | |
|     }
 | |
| 
 | |
|     private void SaveOpenInsightFile(string reportFullPath, DateTime dateTime, List<QS408M.Description> descriptions, Test[] tests)
 | |
|     {
 | |
|         string duplicateFile;
 | |
|         bool isDummyRun = false;
 | |
|         List<(Shared.Properties.IScopeInfo, string)> collection = new();
 | |
|         string duplicateDirectory = Path.Combine(_FileConnectorConfiguration.SourceFileLocation, _CellInstanceName);
 | |
|         bool isStratusQual = _Logistics.MesEntity is "BIORAD4" or "BIORAD5" && _Logistics.MesEntity == _Logistics.ProcessJobID;
 | |
|         if (isStratusQual)
 | |
|             duplicateDirectory = duplicateDirectory.Replace("MET08THFTIRQS408M", "MET08THFTIRSTRATUS");
 | |
|         if (!Directory.Exists(duplicateDirectory))
 | |
|             _ = Directory.CreateDirectory(duplicateDirectory);
 | |
|         string successDirectory = _FileConnectorConfiguration.AlternateTargetFolder;
 | |
|         if (descriptions.Count == 0 || tests.Length == 0)
 | |
|             duplicateFile = Path.Combine(duplicateDirectory, Path.GetFileName(reportFullPath));
 | |
|         else
 | |
|         {
 | |
|             long? subgroupId;
 | |
|             string fileName = Path.GetFileName(reportFullPath);
 | |
|             long breakAfter = dateTime.AddSeconds(_BreakAfterSeconds).Ticks;
 | |
|             long preWait = _FileConnectorConfiguration?.FileHandleWaitTime is null ? dateTime.AddMilliseconds(1234).Ticks : dateTime.AddMilliseconds(_FileConnectorConfiguration.FileHandleWaitTime.Value).Ticks;
 | |
|             if (string.IsNullOrEmpty(descriptions[0].Reactor) || string.IsNullOrEmpty(descriptions[0].PSN))
 | |
|                 subgroupId = null;
 | |
|             else
 | |
|                 (subgroupId, int? _, string _) = FromIQS.GetCommandText(_IqsConnectionString, _Logistics, descriptions[0], breakAfter, preWait);
 | |
|             if (isStratusQual)
 | |
|             {
 | |
|                 string lines = GetLines(_Logistics, descriptions, isStratusQual);
 | |
|                 if (subgroupId is null)
 | |
|                     collection.Add(new(new ScopeInfo(tests[0], _OpenInsightFilePattern), lines));
 | |
|                 else
 | |
|                     collection.Add(new(new ScopeInfo(tests[0], $"{subgroupId.Value} {_OpenInsightFilePattern}"), lines));
 | |
|             }
 | |
|             if (_StaticRuns.TryGetValue(_Logistics.Sequence, out List<WS.Results> wsResults))
 | |
|             {
 | |
|                 if (wsResults is null || wsResults.Count != 1)
 | |
|                     throw new NullReferenceException($"{nameof(wsResults)} {wsResults?.Count} != 1 {_Logistics.Sequence}!");
 | |
|                 lock (_StaticRuns)
 | |
|                     wsResults[0] = WS.Results.Get(wsResults[0], subgroupId);
 | |
|             }
 | |
|             if (!fileName.StartsWith("Viewer"))
 | |
|                 duplicateFile = Path.Combine(duplicateDirectory, $"{subgroupId} {fileName}".TrimStart());
 | |
|             else
 | |
|                 duplicateFile = Path.Combine(duplicateDirectory, $"{$"Viewer {subgroupId}".TrimEnd()} {fileName.Replace("Viewer", string.Empty)}");
 | |
|             string weekOfYear = _Calendar.GetWeekOfYear(_Logistics.DateTimeFromSequence, CalendarWeekRule.FirstDay, DayOfWeek.Sunday).ToString("00");
 | |
|             FromIQS.Save(_OpenInsightApiECDirectory, _Logistics, reportFullPath, descriptions.First(), subgroupId, weekOfYear);
 | |
|         }
 | |
|         if (_IsEAFHosted && _FileConnectorConfiguration.FileScanningIntervalInSeconds > 0)
 | |
|         {
 | |
|             if (!isStratusQual)
 | |
|                 File.Copy(reportFullPath, duplicateFile, overwrite: true);
 | |
|             WaitForFileConsumption(dateTime, descriptions, isDummyRun, successDirectory, duplicateDirectory, collection, duplicateFile);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private Tuple<string, Test[], JsonElement[], List<FileInfo>> GetExtractResult(string reportFullPath, DateTime dateTime)
 | |
|     {
 | |
|         Tuple<string, Test[], JsonElement[], List<FileInfo>> results;
 | |
|         string[] lines = File.ReadAllLines(reportFullPath);
 | |
|         ProcessDataStandardFormat processDataStandardFormat = ProcessDataStandardFormat.GetProcessDataStandardFormat(reportFullPath, lines);
 | |
|         _Logistics = new Logistics(reportFullPath, processDataStandardFormat);
 | |
|         SetFileParameterLotIDToLogisticsMID();
 | |
|         JsonElement[] jsonElements = ProcessDataStandardFormat.GetArray(reportFullPath, lines, processDataStandardFormat);
 | |
|         List<QS408M.Description> descriptions = QS408M.ProcessData.GetDescriptions(jsonElements);
 | |
|         Test[] tests = (from l in descriptions select (Test)l.Test).ToArray();
 | |
|         if (_IsEAFHosted && _FileConnectorConfiguration.FileScanningIntervalInSeconds > 0)
 | |
|             SaveOpenInsightFile(reportFullPath, dateTime, descriptions, tests);
 | |
|         results = new Tuple<string, Test[], JsonElement[], List<FileInfo>>(string.Join(Environment.NewLine, processDataStandardFormat.Logistics[0]), tests, jsonElements, new List<FileInfo>());
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
| } |