Skip to main content

XPath expression to select children of parent in xpath (that works in javascript and in c#)

TLDR: I want to write an xpath expression to select all (at any level) children of parent node using both '..' and 'parent::', that will work in c# and javascript/typescript.

I am limited to xsd 1.0. I do not want to use preceding-sibling axis.

My xml looks like

    <tns:s03.q01 id="s03.q01" text="Select the specific issues encountered (please select all that apply) and indicate the corresponding MBAs" description="" isAnsewerRequired="true" type="Question">
      <tns:choices>
        <tns:multiChoiceAssert condition="count(parent::tns:choices/*[@value='true']) = 0" message="'{$locator}\Choices' must have atleast one choice selected." relevantNodeLocator="string(../../@id)" />
        <tns:s03.q01.c01 value="true" id="s03.q01.c01" text="Bias" description="" type="Choice">
          <tns:selectMBAAssert condition="parent::*[@value='true'] and count(parent::*/tns:mba) = 0" message="'Select MBAs for {$locator}'." relevantNodeLocator="string(../@id)" />
          <tns:selectValueAssert condition="parent::*[@value='false'] and count(parent::*/tns:mba) &gt; 0" message="Select '{$locator}'." relevantNodeLocator="string(../@id)" />
          <tns:mba>m1</tns:mba>
          <tns:mba>m2</tns:mba>
        </tns:s03.q01.c01>
        <tns:s03.q01.c02 id="s03.q01.c02" text="Poor measurement quality" description="" type="Choice" value="false">
          <tns:selectMBAAssert condition="parent::*[@value='true'] and count(parent::*/tns:mba) = 0" message="'Select MBAs for {$locator}'." relevantNodeLocator="string(../@id)" />
          <tns:selectValueAssert condition="parent::*[@value='false'] and count(parent::*/tns:mba) &gt; 0" message="Select '{$locator}'." relevantNodeLocator="string(../@id)" />
        </tns:s03.q01.c02>
        <tns:s03.q01.c03 id="s03.q01.c03" text="Statistically significant MUF" description="" type="Choice" value="false">
          <tns:selectMBAAssert condition="parent::*[@value='true'] and count(parent::*/tns:mba) = 0" message="'Select MBAs for {$locator}'." relevantNodeLocator="string(../@id)" />
          <tns:selectValueAssert condition="parent::*[@value='false'] and count(parent::*/tns:mba) &gt; 0" message="Select '{$locator}'." relevantNodeLocator="string(../@id)" />
        </tns:s03.q01.c03>
      </tns:choices>
      <tns:comment id="s03.q01.comment" title="Comments" description="If the answer to the previous questions is ā€œYes,ā€ please provide the details, including the document type (initial DIQ, DIQ update, initial LOF information, LOF information update), facility/MBA code, the date due and the date received.">
        <tns:s03.q01.comment dateRequested="2023-03-23" documentType="2" facility-mbaCode="m1" />
        <tns:s03.q01.comment dateRequested="2023-04-24" documentType="4" facility-mbaCode="f2" />
      </tns:comment>
    </tns:s03.q01>

I am using c# and javascript and hence I am limited to xsd 1.0 I define validations using nodes like selectMBAAssert and selectValueAssert with rule in the 'condition' attribute. So while parsing the xml I select all nodes with @condition attribute and evaluate the condition in the context of the same node.

The code to do so works fine in c# which looks like

            XmlDocument document = getDocumentWithSchema(xmlTextReader, schemaPath);//this is a method to load document with schema and xml.
            ValidationEventHandler eventHandler = new ValidationEventHandler(ValidationEventHandler);

            var nsmgr = new XmlNamespaceManager(document.NameTable);
            nsmgr.AddNamespace("tns", "http://tempuri.org/XMLSchema2.xsd");

            document.Validate(eventHandler);
            //var nav = document.CreateNavigator();
            var list = document.SelectNodes("//*[@condition]", nsmgr);
            foreach (XmlNode node in list)
            {
                var condition = node.Attributes["condition"].Value;
                var nodeNav = node.CreateNavigator();
                var result = nodeNav.evaluateXPath(condition, nsmgr);
                //If the condition is "//tns:choices" results contains 1 element.
                if ((bool)result)
                {
                    var errorMsg = node.Attributes["message"].Value;
                    var locator = node.Attributes["relevantNodeLocator"].Value;

                    var nodeLocator = nodeNav.evaluateXPath(locator, nsmgr);

                    errorMsg = errorMsg.Replace("{$locator}", nodeLocator.ToString());
                    Console.WriteLine($"Validation error:{errorMsg}");
                }

            }

However code in typescript does work the same way.

  validateXML(questionId: any, xmlData: Document) {


    var conditionNodes = xmlData.evaluate("//*[@id='" + questionId + "']/*[@condition]", xmlData.documentElement, null, XPathResult.ANY_TYPE, null);
    var toretErrors: string = "";
    try {
      var currnetConditionNode: any = conditionNodes.iterateNext();

      while (currnetConditionNode) {
        //console.log(thisNode.textContent);

        var atributeValue = currnetConditionNode.attributes["condition"].nodeValue;
        var validationResult = xmlData.evaluate(atributeValue, currnetConditionNode, this.NSResolver, XPathResult.ANY_TYPE, null)
        console.log(validationResult);
        if (validationResult.booleanValue) {
          toretErrors += currnetConditionNode.attributes["message"].nodeValue;
        }
        currnetConditionNode = conditionNodes.iterateNext();
      }
    }
    catch (e) {
      console.error('Error: Document parsing error ' + e);
    }
    return toretErrors;

  }
  NSResolver(nsPrefix) {
    if (nsPrefix == "tns") {
      return "http://tempuri.org/XMLSchema2.xsd";
    }
    return null;
  }

How may i write the condition in selectMBAAssert node to select all (at any level) parent's children of type tns:mba?

Another question I think relevant to this question, are '..' and 'parent::' interchangeable? How may I script the condition attribute use both of them?

Sorry to make it lengthy question. Please let me know if you need any more details.

Via Active questions tagged javascript - Stack Overflow https://ift.tt/E1W2CjQ

Comments

Popular posts from this blog

ValueError: X has 10 features, but LinearRegression is expecting 1 features as input

So, I am trying to predict the model but its throwing error like it has 10 features but it expacts only 1. So I am confused can anyone help me with it? more importantly its not working for me when my friend runs it. It works perfectly fine dose anyone know the reason about it? cv = KFold(n_splits = 10) all_loss = [] for i in range(9): # 1st for loop over polynomial orders poly_order = i X_train = make_polynomial(x, poly_order) loss_at_order = [] # initiate a set to collect loss for CV for train_index, test_index in cv.split(X_train): print('TRAIN:', train_index, 'TEST:', test_index) X_train_cv, X_test_cv = X_train[train_index], X_test[test_index] t_train_cv, t_test_cv = t[train_index], t[test_index] reg.fit(X_train_cv, t_train_cv) loss_at_order.append(np.mean((t_test_cv - reg.predict(X_test_cv))**2)) # collect loss at fold all_loss.append(np.mean(loss_at_order)) # collect loss at order plt.plot(np.log(al...

Sorting large arrays of big numeric stings

I was solving bigSorting() problem from hackerrank: Consider an array of numeric strings where each string is a positive number with anywhere from to digits. Sort the array's elements in non-decreasing, or ascending order of their integer values and return the sorted array. I know it works as follows: def bigSorting(unsorted): return sorted(unsorted, key=int) But I didnt guess this approach earlier. Initially I tried below: def bigSorting(unsorted): int_unsorted = [int(i) for i in unsorted] int_sorted = sorted(int_unsorted) return [str(i) for i in int_sorted] However, for some of the test cases, it was showing time limit exceeded. Why is it so? PS: I dont know exactly what those test cases were as hacker rank does not reveal all test cases. source https://stackoverflow.com/questions/73007397/sorting-large-arrays-of-big-numeric-stings

How to load Javascript with imported modules?

I am trying to import modules from tensorflowjs, and below is my code. test.html <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</title </head> <body> <script src="https://cdn.jsdelivr.net/npm/@tensorflow/tfjs@2.0.0/dist/tf.min.js"></script> <script type="module" src="./test.js"></script> </body> </html> test.js import * as tf from "./node_modules/@tensorflow/tfjs"; import {loadGraphModel} from "./node_modules/@tensorflow/tfjs-converter"; const MODEL_URL = './model.json'; const model = await loadGraphModel(MODEL_URL); const cat = document.getElementById('cat'); model.execute(tf.browser.fromPixels(cat)); Besides, I run the server using python -m http.server in my command prompt(Windows 10), and this is the error prompt in the console log of my browser: Failed to loa...