好吧,我正在尝试使用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文件添加任何内容。
我认为这个话题很老但我遇到了类似的问题。所以,我只是为那些仍有类似问题的人分享。
在运行此代码时,它会被编译和执行(带有一些警告,但没有任何阻碍执行),但它不会向我的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文件运行您的代码,并且没有任何警告。
您的命令修改了内存中XML的DOM表示,但您错过了将其写回文件。所以添加以下行应该可以解决您的问题:
...
}
// write back to file:
xmlSaveFileEnc("/var/www/db/db_policy.xml", pDoc, "UTF-8");
xmlFreeDoc(pDoc);
xmlCleanupParser();
return (1);