使用libxml2在现有XML中附加数据

问题描述 投票:0回答:2

好吧,我正在尝试使用C编程和libxml2 modulel追加数据,但面临很多问题,因为我对此很新。我的代码旨在首先根据用户输入从XML文件中获取一个元素节点,然后抓取该子节点的父节点并在其中追加另一个子节点。

XML文件:

<policyList>
        <policySecurity>
                <policyName>AutoAdd</policyName>
                <deviceName>PA-722</deviceName>
                <status>ACTIVE</status>
                <srcZone>any</srcZone>
                <dstZone>any</dstZone>
                <srcAddr>5.5.5.5</srcAddr>
                <dstAddr>5.5.5.4</dstAddr>
                <srcUser>any</srcUser>
                <application>any</application>
                <service>htds</service>
                <urlCategory>any</urlCategory>
                <action>deny</action>
        </policySecurity>
        <policySecurity>
                <policyName>Test-1</policyName>
                <deviceName>PA-710</deviceName>
                <status>ACTIVE</status>
                <srcZone>any</srcZone>
                <dstZone>any</dstZone>
                <srcAddr>192.168.1.23</srcAddr>
                <dstAddr>8.8.8.8</dstAddr>
                <srcUser>vivek</srcUser>
                <application>any</application>
                <service>http</service>
                <urlCategory>any</urlCategory>
                <action>deny</action>
        </policySecurity>
</policyList>

C代码:

int main(){
xmlDocPtr pDoc = xmlReadFile("/var/www/db/db_policy.xml", NULL, XML_PARSE_NOBLANKS | XML_PARSE_NOERROR | XML_PARSE_NOWARNING | XML_PARSE_NONET);
    if (pDoc == NULL)
    {
        fprintf(stderr, "Document not parsed successfully.\n");
        return 0;
    }

root_element = xmlDocGetRootElement(pDoc);
if (root_element == NULL)
 {
        fprintf(stderr, "empty document\n");
        xmlFreeDoc(pDoc);
        return 0;
    }

printf("Root Node is %s\n", root_element->name);



        xmlChar* srcaddr = "5.5.5.5";
        xmlChar *xpath = (xmlChar*) "//srcAddr";
        xmlNodeSetPtr nodeset;
        xmlXPathObjectPtr result;
        int i;
        xmlChar *keyword;
        xmlXPathContextPtr context;
        xmlNodePtr resdev;
        xmlChar* resd;
         context = xmlXPathNewContext(pDoc);
        if (context == NULL) {
                printf("Error in xmlXPathNewContext\n");
        }
        result = xmlXPathEvalExpression(xpath, context);
        xmlXPathFreeContext(context);
        if (result == NULL) {
                printf("Error in xmlXPathEvalExpression\n");
        }
         if(xmlXPathNodeSetIsEmpty(result->nodesetval)){
                xmlXPathFreeObject(result);
                printf("No result\n");
        };

        if (result) {
                nodeset = result->nodesetval;
                for (i=0; i < nodeset->nodeNr; i++) {
                        keyword = xmlNodeListGetString(pDoc, nodeset->nodeTab[i]->xmlChildrenNode, 1);
                printf("keyword: %s\n", keyword);
                if(strcmp(keyword, srcaddr) == 0){
                        xmlNodePtr pNode = xmlNewNode(0, (xmlChar*)"service");
                        xmlNodeSetContent(pNode, (xmlChar*)"nonser");
                        xmlAddSibling(result, pNode);
                        printf("added");
                        }
                xmlFree(keyword);
                }
                xmlXPathFreeObject (result);
        }
        xmlFreeDoc(pDoc);
        xmlCleanupParser();
        return (1);
}

在运行此代码时,它会被编译和执行(带有一些警告,但没有任何阻碍执行),但它不会向我的XML文件添加任何内容。

c xml libxml2
2个回答
1
投票

我认为这个话题很老但我遇到了类似的问题。所以,我只是为那些仍有类似问题的人分享。

在运行此代码时,它会被编译和执行(带有一些警告,但没有任何阻碍执行),但它不会向我的XML文件添加任何内容。

首先:在我看来,C中的警告比错误更糟糕,因为它可以让你运行错误的代码。所以,我的第一个建议是不要忽视这些警告(虽然我无论如何都不能建议任何人)。

第二:当我运行这段代码时,我看到一个有意义的警告:

> warning: passing argument 1 of ‘xmlAddSibling’ from incompatible
> pointer type [-Wincompatible-pointer-types]
> 
> note: expected ‘xmlNodePtr {aka struct _xmlNode *}’ but argument is of
> type ‘xmlXPathObjectPtr {aka struct _xmlXPathObject *}’

当您从http://www.xmlsoft.org/html/libxml-tree.html查看xmlAddSibling时,您可以看到:

xmlNodePtr  xmlAddSibling       (xmlNodePtr cur, xmlNodePtr elem)

这意味着两个参数的类型都应该是xmlNodePtr。但是,“result”具有xmlXPathObjectPtr的类型,这意味着指针类型完全不同。您真正想要做的是根据您比较的字符串将子项添加到您找到的父项:(if(strcmp(keyword,srcaddr)== 0))。

所以你找到父母的方式是完全正确的。但是有两个问题:首先你永远不会更新“结果”(如果我们假设你想象“结果”是不正确的父),因为“nodeset-> nodeTab [i]”处于一个永远不会放任何东西的for循环中结果”。第二个问题是即使你基于“nodeset-> nodeTab [i]”更新了“结果”,它们仍然有不同类型的指针(如前所述)。因此,您必须将xmlAddSibling用于正确的父级并使用正确的指针类型。正如您在下面所看到的,“nodeTab”具有我们正在寻找的“xmlNodePtr”类型,“nodeset-> nodeTab [i]”是父类。

Structure xmlNodeSet
struct _xmlNodeSet {
    int nodeNr  : number of nodes in the set
    int nodeMax : size of the array as allocated
    > `xmlNodePtr * nodeTab : array of nodes in no particular order`

}

所以你应该改变:

xmlAddSibling(result,pNode);

至:

xmlAddSibling(nodeset-> nodeTab [i],pNode);

最后:你没有保存更改。所以,通过添加来保存它

xmlSaveFileEnc(“note.xml”,pDoc,“UTF-8”);

之前

xmlFreeDoc释放(PDOC);

通过这些更改,我能够使用您的XML文件运行您的代码,并且没有任何警告。


0
投票

您的命令修改了内存中XML的DOM表示,但您错过了将其写回文件。所以添加以下行应该可以解决您的问题:

   ...
}

// write back to file:
xmlSaveFileEnc("/var/www/db/db_policy.xml", pDoc, "UTF-8");

xmlFreeDoc(pDoc);
xmlCleanupParser();
return (1);
© www.soinside.com 2019 - 2024. All rights reserved.