XSLT: извлечь набор элементов из файла XML и получить значения из файла TXT - PullRequest
0 голосов
/ 12 сентября 2018

У меня есть два файла: файл Rules_Generic (generic.xml) и (Rules_Values.txt).Я хочу найти способ написать новый XML-файл, извлекая синтаксис Rules из файла generic.xml и заполняя значения из Rules_Values.txt

Я искал и обнаружил, что XSLT можно использовать.следующее (Rules_Values.txt)

#VNF_Type   VNF_Id    min_num_Instance  max_num_Instance    ScalingOut%     ScalingIn%      min_num_CPUs        max_num_CPUs    Migration_Type  

Firewall     f1             1                   4                   60%            20%              4               8               Pre-Copy
Firewall     f2             2                   6                   80%            10%              2               6               Post-Copy   
Cache        c1             2                   8                   90%            30%              2               6               Hybrid

следующее (generic.xml)

<?xml version="1.0" encoding="UTF-8"?>
<PolicySet  PolicySetId="PS2" PolicySetName="ResourceAdaptation" Client_Id =" ">
    <PolicySet SFC_Name="">
        <Policy PolicyId=" ">
            <Target>
                <Subject>VNF_Type</Subject>     
            </Target>
    <!-- Horizontal_Scaling -->         
            <Rule RuleId="R1" Effect="Affinity">
                <Target>
                    <Subject>VNF_ID</Subject>
                    <Action>Horizontal_Scaling</Action>             
                    <Resource>numOfInstances</Resource>
                </Target>
            <conditions>
                <condition>min_numOfInstances: Any_Positive_Integer</condition>
                <condition>max_numOfInstances: Any_Positive_Integer</condition>     
            </conditions>           
            </Rule>
    <!-- Vertical_Scaling -->   
            <Rule RuleId="R2" Effect="Affinity">
                <Target>
                    <Subject>VNF_ID</Subject>
                    <Action>Vertical_Scaling</Action>               
                    <Resource>numOfCPUcores</Resource>
                </Target>
            <conditions>
                <condition>min_numOfCPUcores: Any_Positive_Integer</condition>
                <condition>max_numOfCPUcores: Any_Positive_Integer</condition>  
            </conditions>           
            </Rule> 
    <!-- Horizontal-Scaling threshold-based 1- scaling_Out 2- scaling_In -->        
            <Rule RuleId = "R3" Effect="Affinity">
                <Target>
                    <Subject>VNF_ID</Subject>               
                    <Action>Scaling_Out</Action>                
                    <Resource>cpu_Utilization</Resource>
                </Target>
            <conditions>
                <condition>utilization greater_than threshold%</condition>                      
            </conditions>           
            </Rule>
            <Rule RuleId = "R4" Effect="Affinity">
                <Target>
                    <Subject>VNF_ID</Subject>                               
                    <Action>Scaling_In</Action>             
                    <Resource>cpu_Utilization</Resource>
                </Target>
            <conditions>
                <condition>utilization less_than threshold%</condition>                     
            </conditions>           
            </Rule>
    <!-- For Migration : specify 1. MigrationType 2.  -->           
            <Rule RuleId = "R7" Effect="Affinity">
                <Target>
                    <Subject>VNF_ID</Subject>                                                       
                    <Action>Migration</Action>              
                </Target>
            <conditions>
                <condition>MigrationType: [Pre-Copy , Post-Copy, Hybrid]</condition>
            </conditions>           
            </Rule>

        </Policy>   
    </PolicySet>


</PolicySet>

И желаемый вывод (out.xml)

<?xml version="1.0" encoding="UTF-8"?>
<PolicySet  PolicySetId="PS2" PolicySetName="ResourceAdaptation" Client_Id =" ">
    <PolicySet SFC_Name="">
        <Policy PolicyId="p1">
            <Target>
                <Subject>Firewall</Subject>     
            </Target>
            <Rule RuleId="R1" Effect="Affinity">
                <Target>
                    <Subject>f1</Subject>
                    <Action>Horizontal_Scaling</Action>             
                    <Resource>numOfInstances</Resource>
                </Target>
            <conditions>
                <condition>min_numOfInstances: 1</condition>
                <condition>max_numOfInstances: 4</condition>    
            </conditions>           
            </Rule>
            <Rule RuleId="R2" Effect="Affinity">
                <Target>
                    <Subject>f1</Subject>
                    <Action>Vertical_Scaling</Action>               
                    <Resource>numOfCPUcores</Resource>
                </Target>
            <conditions>
                <condition>min_numOfCPUcores: 4</condition>
                <condition>max_numOfCPUcores: 8</condition>     
            </conditions>           
            </Rule> 
            <Rule RuleId = "R3" Effect="Affinity">
                <Target>
                    <Subject>f1</Subject>               
                    <Action>Scaling_Out</Action>                
                    <Resource>cpu_Utilization</Resource>
                </Target>
            <conditions>
                <condition>utilization greater_than 60%</condition>                     
            </conditions>           
            </Rule>
            <Rule RuleId = "R4" Effect="Affinity">
                <Target>
                    <Subject>f1</Subject>                               
                    <Action>Scaling_In</Action>             
                    <Resource>cpu_Utilization</Resource>
                </Target>
            <conditions>
                <condition>utilization less_than 20%</condition>                        
            </conditions>           
            </Rule>
            <Rule RuleId = "R5" Effect="Affinity">
                <Target>
                    <Subject>f1</Subject>                                                       
                    <Action>Migration</Action>              
                </Target>
            <conditions>
                <condition>MigrationType: Pre-Copy</condition>
            </conditions>           
            </Rule>
            <Rule RuleId="R6" Effect="Affinity">
                <Target>
                    <Subject>f2</Subject>
                    <Action>Horizontal_Scaling</Action>             
                    <Resource>numOfInstances</Resource>
                </Target>
            <conditions>
                <condition>min_numOfInstances: 2</condition>
                <condition>max_numOfInstances: 6</condition>    
            </conditions>           
            </Rule>
            <Rule RuleId="R7" Effect="Affinity">
                <Target>
                    <Subject>f2</Subject>
                    <Action>Vertical_Scaling</Action>               
                    <Resource>numOfCPUcores</Resource>
                </Target>
            <conditions>
                <condition>min_numOfCPUcores: 2</condition>
                <condition>max_numOfCPUcores: 6</condition>     
            </conditions>           
            </Rule> 
            <Rule RuleId = "R8" Effect="Affinity">
                <Target>
                    <Subject>f2</Subject>               
                    <Action>Scaling_Out</Action>                
                    <Resource>cpu_Utilization</Resource>
                </Target>
            <conditions>
                <condition>utilization greater_than 80%</condition>                     
            </conditions>           
            </Rule>
            <Rule RuleId = "R9" Effect="Affinity">
                <Target>
                    <Subject>f2</Subject>                               
                    <Action>Scaling_In</Action>             
                    <Resource>cpu_Utilization</Resource>
                </Target>
            <conditions>
                <condition>utilization less_than 10%</condition>                        
            </conditions>           
            </Rule>
            <Rule RuleId = "R10" Effect="Affinity">
                <Target>
                    <Subject>f1</Subject>                                                       
                    <Action>Migration</Action>              
                </Target>
            <conditions>
                <condition>MigrationType: Post-Copy</condition>
            </conditions>           
            </Rule>                                 
        </Policy>
        <Policy PolicyId="p2">
            <Target>
                <Subject>Cache</Subject>        
            </Target>
            <Rule RuleId="R11" Effect="Affinity">
                <Target>
                    <Subject>c1</Subject>
                    <Action>Horizontal_Scaling</Action>             
                    <Resource>numOfInstances</Resource>
                </Target>
            <conditions>
                <condition>min_numOfInstances: 2</condition>
                <condition>max_numOfInstances: 8</condition>    
            </conditions>           
            </Rule>
            <Rule RuleId="R12" Effect="Affinity">
                <Target>
                    <Subject>c1</Subject>
                    <Action>Vertical_Scaling</Action>               
                    <Resource>numOfCPUcores</Resource>
                </Target>
            <conditions>
                <condition>min_numOfCPUcores: 2</condition>
                <condition>max_numOfCPUcores: 6</condition>     
            </conditions>           
            </Rule> 
            <Rule RuleId = "R13" Effect="Affinity">
                <Target>
                    <Subject>c1</Subject>               
                    <Action>Scaling_Out</Action>                
                    <Resource>cpu_Utilization</Resource>
                </Target>
            <conditions>
                <condition>utilization greater_than 90%</condition>                     
            </conditions>           
            </Rule>
            <Rule RuleId = "R14" Effect="Affinity">
                <Target>
                    <Subject>c1</Subject>                               
                    <Action>Scaling_In</Action>             
                    <Resource>cpu_Utilization</Resource>
                </Target>
            <conditions>
                <condition>utilization less_than 30%</condition>                        
            </conditions>           
            </Rule>
            <Rule RuleId = "R15" Effect="Affinity">
                <Target>
                    <Subject>f1</Subject>                                                       
                    <Action>Migration</Action>              
                </Target>
            <conditions>
                <condition>MigrationType: Hybrid</condition>
            </conditions>           
            </Rule> 
        </Policy>           
    </PolicySet>


</PolicySet>

Мой код как следующий (.java):

    public static void main(String[] args) throws SaxonApiException {

        Processor proc = new Processor(false);
        XsltCompiler comp = proc.newXsltCompiler();
        XsltExecutable exp = comp.compile(new StreamSource(new File(
                "C:/Users/eclipse-workspace/PolicyFiles_Generator_V2/src/transformation.xsl")));
        Serializer out = proc.newSerializer();
        out.setOutputProperty(Serializer.Property.METHOD, "xml");
        out.setOutputProperty(Serializer.Property.INDENT, "yes");
        out.setOutputFile(new File("output.xml"));
        XsltTransformer trans = exp.load();
        trans.setInitialTemplate(new QName("main"));
        trans.setDestination(out);
        trans.transform();
}

.xsl код:

<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:output omit-xml-declaration="yes" indent="yes"/>
    <xsl:variable name="vText" select="tokenize(unparsed-text('file:///C:/Users/eclipse-workspace/PolicyFiles_Generator_V2/src/Rules_Values.txt'),'&#xa;')"/>

  <xsl:template  match="/">
    <shirts>
      <xsl:apply-templates select="document('C:/Users/Hsuwi/eclipse-workspace/PolicyFiles_Generator_V2/src/GenericTemplate.xml')"/>
      <xsl:apply-templates/>
    </shirts>
  </xsl:template>

  <xsl:template match="@*|node()">
    <xsl:copy>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
  </xsl:template>

    <xsl:template name="main" match="/">
        <items>
            <xsl:for-each select="$vText">
                <item>
                   ...........
            </xsl:for-each>
        </items>
    </xsl:template>
</xsl:stylesheet>

Я не знаю, как завершить код xsl, чтобы извлечьПравило из 1-го XML-файла и заполните каждое Правило значениями из .txt файла

Любой совет?Помогите ??или если есть какой-либо другой способ достичь цели.Файл .txt является гибким (что означает, что я могу переупорядочить элементы)

1 Ответ

0 голосов
/ 12 сентября 2018

В вашем вопросе слишком много и слишком разнообразных выборочных данных для точного ответа, но, как сказано в комментарии, мне интересно, подходит ли подход, который определяет именованный шаблон или функцию, принимая число значений, которые вы хотите заполнить как именованные аргументы, например

  <xsl:function name="mf:fill-template" as="element(Policy)">
      <xsl:param name="p1"/>
      <xsl:param name="p2"/>
      <xsl:param name="p3"/>
      <Policy xsl:expand-text="yes">
          <Rule>
              <foo>test {$p2}</foo>
          </Rule>
          <Rule att="{$p1}">
              ...
          </Rule>
          <Rule>
              <header>...</header>
              <value>... {$p3}</value>
          </Rule>
      </Policy>
  </xsl:function>

, а затем при необходимости вызывает эту функцию для каждой строки в вашем тексте, например

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    xmlns:mf="http://example.com/mf"
    exclude-result-prefixes="#all"
    version="3.0">


  <xsl:function name="mf:fill-template" as="element(Policy)">
      <xsl:param name="p1"/>
      <xsl:param name="p2"/>
      <xsl:param name="p3"/>
      <Policy xsl:expand-text="yes">
          <Rule>
              <foo>test {$p2}</foo>
          </Rule>
          <Rule att="{$p1}">
              ...
          </Rule>
          <Rule>
              <header>...</header>
              <value>... {$p3}</value>
          </Rule>
      </Policy>
  </xsl:function>

  <xsl:param name="data" as="xs:string">P1  P2  P3
a b c
c d f
g h i</xsl:param>

  <xsl:output method="xml" indent="yes" />

  <xsl:template match="/">
    <PolicySet>
        <xsl:for-each select="tail(tokenize($data, '\r?\n'))">
            <xsl:sequence select="let $args := tokenize(., '\s+') return mf:fill-template($args[1], $args[2], $args[3])"/>
        </xsl:for-each>
    </PolicySet>
  </xsl:template>

</xsl:stylesheet>

В примере для полноты и компактности используется строковый параметр, но вы, конечно, можете использовать <xsl:param name="data" select="unparsed-text('file.txt')"/> вместо.

Пример в сети: https://xsltfiddle.liberty -development.net / bFDb2CH .

Не уверен, поможет ли это, но вставка этого предложения в комментарий будетдовольно сложно.

Я понимаю, что я не обработал ваш входной XML, а скорее предположил, что вы можете адаптировать его и создать функцию с его конструктором шаблона / последовательности.Возможно, это можно было бы автоматизировать, но в настоящее время я не вижу, как сопоставить различные имена столбцов с различными и переменными позициями текста в примере XML.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...