mirror of
				https://github.com/Theodor-Springmann-Stiftung/hamann-ausgabe-core.git
				synced 2025-10-31 02:05:33 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			183 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			183 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| namespace HaWeb.XMLTests;
 | |
| using HaWeb.Models;
 | |
| using System.Linq;
 | |
| using System.Xml;
 | |
| using System.Xml.Linq;
 | |
| using System.Xml.XPath;
 | |
| 
 | |
| public class XMLTester {
 | |
|     private Dictionary<string, INodeRule>? _Ruleset;
 | |
|     private Dictionary<string, ICollectionRule>? _CollectionRuleset;
 | |
|     private List<XMLRootDocument>? _Documents;
 | |
|     private Dictionary<string, HashSet<string>>? _IDs;
 | |
|     private Dictionary<string, HashSet<string>>? _CollectionIDs;
 | |
|     private Dictionary<string, List<(XElement, XMLRootDocument)>?> _XPathEvaluated;
 | |
|     public XMLTester (IXMLTestService testService, Dictionary<string, Models.FileList?>? filelists) {
 | |
|         _Ruleset  = testService.Ruleset;
 | |
|         _CollectionRuleset = testService.CollectionRuleset;
 | |
|         if (filelists != null) {
 | |
|             foreach (var fl in filelists) {
 | |
|                 if (fl.Value != null)  {
 | |
|                     if (_Documents == null) _Documents = new List<XMLRootDocument>();
 | |
|                     var docs = fl.Value.GetFileList();
 | |
|                     if (docs != null) _Documents.AddRange(docs);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         _XPathEvaluated = new Dictionary<string, List<(XElement, XMLRootDocument)>?>();
 | |
|     }
 | |
| 
 | |
|     public void Test() {
 | |
|         if (_Ruleset == null) return;        
 | |
|         _IDs = new System.Collections.Generic.Dictionary<string, HashSet<string>>();
 | |
|         foreach (var rule in _Ruleset) {
 | |
|             buildIDs(rule.Value);
 | |
|             checkRequiredAttributes(rule.Value);
 | |
|             checkReferences(rule.Value);
 | |
|         }
 | |
|         if (_CollectionRuleset == null) return;
 | |
|         _CollectionIDs = new Dictionary<string, HashSet<string>>();
 | |
|         foreach (var collectionrule in _CollectionRuleset) {
 | |
|             buildIDs(collectionrule.Value);
 | |
|             checkReferences(collectionrule.Value);
 | |
|         }
 | |
|     }
 | |
|     private void checkReferences(INodeRule rule) {
 | |
|         if (rule.References == null || !rule.References.Any()) return;
 | |
|         var elements = GetEvaluateXPath(rule.XPath);
 | |
|         if (elements != null && elements.Any()) {
 | |
|             foreach (var e in elements) {
 | |
|                 foreach (var r in rule.References) {
 | |
|                     var hasattr = checkAttribute(e.Item1, r.LinkAttribute, e.Item2, false);
 | |
|                     var keyname = r.RemoteElement + "-" + r.RemoteAttribute;
 | |
|                     if (_IDs != null && _IDs.ContainsKey(keyname) && hasattr) {
 | |
|                         var val = e.Item1.Attribute(r.LinkAttribute)!.Value;
 | |
|                         if (!_IDs[keyname].Contains(val)) {
 | |
|                             e.Item2.Log(generateLogMessage(e.Item1) + "Verlinktes Element " + val + " nicht gefunden.");
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void checkReferences(ICollectionRule rule) {
 | |
|         if (rule.Backlinks == null || !rule.Backlinks.Any() || !_CollectionIDs.ContainsKey(rule.Name)) return;
 | |
|         foreach (var bl in rule.Backlinks) {
 | |
|             var elemens = GetEvaluateXPath(bl);
 | |
|             if (elemens != null && elemens.Any()) {
 | |
|                 foreach(var r in rule.GenerateBacklinkString(elemens)) {
 | |
|                     if (!r.Item4 && !_CollectionIDs[rule.Name].Contains(r.Item1)) {
 | |
|                         r.Item3.Log(generateLogMessage(r.Item2) + "Verlinktes Element " + r.Item1 + " nicht gefunden.");
 | |
|                     }
 | |
|                     if (r.Item4) {
 | |
|                         var coll = _CollectionIDs[rule.Name];
 | |
|                         var items = r.Item1.Split('-');
 | |
|                         var searchterm = items[0];
 | |
|                         var found = coll.Where(x => x.StartsWith(searchterm));
 | |
|                         if (items[0] == "NA" || found == null || !found.Any()) {
 | |
|                             r.Item3.Log(generateLogMessage(r.Item2) + "Verlinktes Element " + r.Item1 + " nicht gefunden.");
 | |
|                         } else {
 | |
|                             for (var i = 1; i < items.Length; i++) {
 | |
|                                 if (items[i] == "NA") break;
 | |
|                                 else {
 | |
|                                     searchterm = searchterm + "-" + items[i];
 | |
|                                     found = found.Where(x => x.StartsWith(searchterm));
 | |
|                                     if (found == null || !found.Any())
 | |
|                                         r.Item3.Log(generateLogMessage(r.Item2) + "Verlinktes Element " + r.Item1 + " nicht gefunden.");
 | |
|                                 }
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void checkRequiredAttributes(INodeRule rule) {
 | |
|         if (rule.Attributes == null) return;
 | |
|         var elements = GetEvaluateXPath(rule.XPath);
 | |
|         if (elements != null && elements.Any()) {
 | |
|             foreach (var e in elements) {
 | |
|                 foreach (var attr in rule.Attributes) {
 | |
|                     checkAttribute(e.Item1, attr, e.Item2);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void buildIDs(INodeRule rule) {
 | |
|         if (!String.IsNullOrWhiteSpace(rule.uniquenessAttribute)) {
 | |
|             checkUniqueness(rule.XPath, rule.uniquenessAttribute);
 | |
|         }
 | |
|         if (rule.References != null && rule.References.Any()) {
 | |
|             foreach (var reference in rule.References) {
 | |
|                 checkUniqueness(reference.RemoteElement, reference.RemoteAttribute);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void buildIDs(ICollectionRule rule) {
 | |
|         if (rule.Bases != null && rule.Bases.Any()) {
 | |
|             var hs = new HashSet<string>();
 | |
|             foreach (var b in rule.Bases) {
 | |
|                 var elemens = GetEvaluateXPath(b);
 | |
|                 if (elemens != null && elemens.Any()) {
 | |
|                     foreach (var r in rule.GenerateIdentificationStrings(elemens)) {
 | |
|                         if (!hs.Add(r.Item1)) {
 | |
|                             r.Item3.Log(generateLogMessage(r.Item2) + "Brief-Seite-Zeile  " + r.Item1 + " mehrdeutig.");
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             _CollectionIDs!.TryAdd(rule.Name, hs);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void checkUniqueness(string xpathelement, string attribute) {
 | |
|         if (_Documents == null || _IDs == null || _IDs.ContainsKey(xpathelement + "-" + attribute)) return;
 | |
|         var hs = new HashSet<string>();
 | |
|         var elements = GetEvaluateXPath(xpathelement);
 | |
|         if (elements != null)
 | |
|             foreach (var e in elements) {
 | |
|                 if (checkAttribute(e.Item1, attribute, e.Item2)) {
 | |
|                     if (!hs.Add(e.Item1.Attribute(attribute)!.Value)) {
 | |
|                         e.Item2.Log(generateLogMessage(e.Item1) + "Attributwert " + e.Item1.Attribute(attribute)!.Value + " doppelt.");
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             _IDs.TryAdd(xpathelement  + "-" + attribute, hs);
 | |
|     }
 | |
| 
 | |
|     private bool checkAttribute(XElement element, string attributename, XMLRootDocument doc, bool log = true) {
 | |
|         if (!element.HasAttributes || element.Attribute(attributename) == null) {
 | |
|             if (log) doc.Log(generateLogMessage(element) + "Attribut " + attributename + " fehlt.");
 | |
|             return false;
 | |
|         }
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     private string generateLogMessage(XElement element) {
 | |
|         return "Zeile " + 
 | |
|             ((IXmlLineInfo)element).LineNumber.ToString() + 
 | |
|             ", Element " +
 | |
|             element.Name +
 | |
|             ": ";
 | |
|     }
 | |
| 
 | |
|     private List<(XElement, XMLRootDocument)>? GetEvaluateXPath(string xpath) {
 | |
|         if (_XPathEvaluated.ContainsKey(xpath)) return _XPathEvaluated[xpath];
 | |
|         if (!_XPathEvaluated.ContainsKey(xpath)) _XPathEvaluated.Add(xpath, null);
 | |
|         if (_Documents == null) return null;
 | |
|         foreach (var d in _Documents) {
 | |
|             var elements = d.GetElement().XPathSelectElements(xpath).ToList();
 | |
|             if (elements != null && elements.Any()) {
 | |
|                 if (_XPathEvaluated[xpath] == null) _XPathEvaluated[xpath] = new List<(XElement, XMLRootDocument)>();
 | |
|                 foreach (var res in elements) {
 | |
|                     _XPathEvaluated[xpath]!.Add((res, d));
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         return _XPathEvaluated[xpath];
 | |
|     }
 | |
| } | 
