using System; using System.Collections.Generic; using System.Runtime.Serialization; using System.Xml; using Aitex.Core.RT.Log; using Aitex.Core.RT.RecipeCenter; using MECF.Framework.Common.Equipment; using SciChart.Core.Extensions; namespace MECF.Framework.Common.Jobs { public enum SequenceLLInOutPath { AInBOut, BInAOut, DInDOut, AInAOut, BInBOut, } [Serializable] [DataContract] public class SequenceInfo { [DataMember] public List Steps { get; set; } [DataMember] public string Name { get; set; } [DataMember] public string WTWCleanRecipe { get; set; } [DataMember] public List PMs { get; set; } [DataMember] public string ThicknessType { get; set; } [DataMember] public Guid InnerId { get; set; } [DataMember] public SequenceLLInOutPath LLInOutPath { get; set; } public SequenceInfo(string name) { Name = name; InnerId = Guid.NewGuid(); LLInOutPath = SequenceLLInOutPath.DInDOut; Steps = new List(); } public string GetRecipe(ModuleName pm) { string attr = $"{pm}Recipe"; foreach (var step in Steps) { if(step.StepModules.Contains(pm)) { return (string)step.StepParameter[attr]; } } return string.Empty; } } public class SequenceInfoHelper { public static SequenceInfo GetInfo(string seqFile) { SequenceInfo info = new SequenceInfo(seqFile); info.PMs = new List(); string content = RecipeFileManager.Instance.GetSequenceAndTryAppendLL(seqFile, false); if (!string.IsNullOrEmpty(content)) { try { XmlDocument dom = new XmlDocument(); dom.LoadXml(content); XmlNodeList lstStepNode = dom.SelectNodes("Aitex/TableSequenceData/Step"); if (lstStepNode == null) { //LOG.Error($"{seqFile} has no step"); return null; } var nodeData = dom.SelectSingleNode("Aitex/TableSequenceData"); if (nodeData != null) { var node = nodeData as XmlElement; info.ThicknessType = node.GetAttribute("ThicknessType"); } foreach (var nodeModelChild in lstStepNode) { XmlElement nodeStep = nodeModelChild as XmlElement; SequenceStepInfo stepInfo = new SequenceStepInfo(); foreach (XmlAttribute attr in nodeStep.Attributes) { stepInfo.StepParameter[attr.Name] = attr.Value; if (attr.Name == "Position" || attr.Name=="LLSelection" || attr.Name == "PMSelection") { if (attr.Value == "LL" || attr.Value=="PM") continue; string[] pos = attr.Value.Split(','); if (pos.Length < 1) { LOG.Write(eEvent.WARN_SEQUENCE, ModuleName.System, $"{seqFile} Position {attr.Value} can not be empty"); return null; } foreach (var po in pos) { if (po.IsEmpty()) continue; if(attr.Name == "PMSelection") { info.PMs.Add(ModuleHelper.Converter(po)); } if (po == "Aligner") { stepInfo.StepModules.AddIfNotContains(ModuleName.Aligner1); stepInfo.StepModules.AddIfNotContains(ModuleName.Aligner2); continue; } ModuleName module = ModuleHelper.Converter(po); if (module == ModuleName.System) { //LOG.Error($"{seqFile} Position {po} not valid"); return null; } stepInfo.StepModules.Add(module); } continue; } if (attr.Name == "AlignerAngle") { if (!double.TryParse(attr.Value, out double angle)) { //LOG.Error($"{seqFile} AlignAngle {attr.Value} not valid"); return null; } stepInfo.AlignAngle = angle; continue; } if(attr.Name == "WTWClean") { info.WTWCleanRecipe = attr.Value; } } info.Steps.Add(stepInfo); } } catch (Exception ex) { LOG.WriteExeption(ex); return null; } } return info; } } }