postgres-修改xml(MySql UpdateXml替代方案)

postgres-修改xml(MySql UpdateXml替代方案),mysql,sql,xml,postgresql,Mysql,Sql,Xml,Postgresql,我正在将mysql数据库迁移到postgres,在一些基本的xml功能方面遇到了障碍。在MySql中,我有一些存储过程可以替换xml文档中的节点,但在postgres中找不到任何方法 以下是我从mysql存储的过程: CREATE DEFINER=`root`@`localhost` PROCEDURE `SP_UpdateExamFilesXmlNode`(examFileId int, xPathExpression varchar(128), xmlNode longtext) BEGIN

我正在将mysql数据库迁移到postgres,在一些基本的xml功能方面遇到了障碍。在MySql中,我有一些存储过程可以替换xml文档中的节点,但在postgres中找不到任何方法

以下是我从mysql存储的过程:

CREATE DEFINER=`root`@`localhost` PROCEDURE `SP_UpdateExamFilesXmlNode`(examFileId int, xPathExpression varchar(128), xmlNode longtext)
BEGIN
    DECLARE xmlData longtext;
    DECLARE newXmlData longtext;
    DECLARE xmlNodeCount int;
    SET xmlData = NULL;
    SELECT xml_data INTO xmlData FROM sonixhub.exam_files WHERE id = examFileId;

    IF xmlData IS NOT NULL THEN
        -- check if the node already exists and if it does then simply update it
        SET xmlNodeCount = ExtractValue(xmlData, CONCAT('count(',xPathExpression,')'));
        IF xmlNodeCount > 0 THEN
            SET newXmlData = UpdateXML(xmlData, xPathExpression, xmlNode);
        -- if node doesn't exist then we have to add it manually
        ELSE
            SET newXmlData = REPLACE(xmlData, '</ImageXmlData>', CONCAT(xmlNode, '</ImageXmlData>'));
        END IF;
        UPDATE sonixhub.exam_files SET xml_data = newXmlData WHERE id = examFileId;
    ELSE
        -- there is no xml found so create xml from scratch and insert the node
        SET xmlData = CONCAT('<ImageXmlData>',xmlNode,'</ImageXmlData>');
        UPDATE sonixhub.exam_files SET xml_data = xmlData WHERE id = examFileId;
    END IF;
END
有没有办法在postgres函数中复制此功能,而不是将逻辑移到应用程序本身

编辑-找到问题的解决方案

我找到了一个混合使用PostgresXML和字符串格式化函数的解决方案

examFileId用于查找要使用xml更新的行,使用表信息更改代码 在我的例子中是硬编码的根节点,但是您可以将其更改为任何您喜欢的

下面是调用函数的方式:

-- this adds <DicomTags> node to your xml value in the table, if <DicomTags> already exists then it's replaced by the one passed in
select update_exam_files_xml_node(1, '/ImageXmlData/DicomTags', '<DicomTags><DicomTag>xxx</DicomTag></DicomTags>');

-- this adds <Settings> node to your xml value in the table, if <Settings> already exists then it's replaced by the one passed in
select update_exam_files_xml_node(1, '/ImageXmlData/Settings', '<Settings>asdf</Settings>');


CREATE OR REPLACE FUNCTION update_exam_files_xml_node(examFileId int, xPathExpression text, xmlNode text)
  RETURNS void AS
$BODY$
    DECLARE xmlData xml;
    DECLARE newXmlData xml;
    DECLARE xmlNodeCount int;
    DECLARE replaceTag text;
BEGIN
    SELECT xml_data INTO xmlData FROM exam_files WHERE id = examFileId;

    IF xml_is_well_formed(xmlNode) = false THEN
        PERFORM add_error_log('update_exam_files_xml_node', 'xmlNode is not well formed xml');
        RETURN;
    END IF;

    IF xmlData IS NOT NULL THEN
        -- check if the node already exists and if it does then simply update it
        IF xmlexists(xPathExpression PASSING BY REF xml(xmlData)) = true THEN
            -- get the node name
            replaceTag := regexp_replace(xPathExpression, '/.*/', '');

            -- replace the existing node with the newly passed in node
            newXmlData := xml(regexp_replace(xmlData::text, '<'||replaceTag||'>.*</'||replaceTag||'>', xmlNode));

        -- if node doesn't exist then we have to add it manually
        ELSE
            newXmlData := xml(REPLACE(xmlData::text, '</ImageXmlData>', xmlNode||'</ImageXmlData>'));
        END IF;
        UPDATE exam_files SET xml_data = newXmlData WHERE id = examFileId;
    ELSE
        -- there is no xml found so create xml from scratch and insert the node
        xmlData := '<ImageXmlData>'||xmlNode||'</ImageXmlData>';
        UPDATE exam_files SET xml_data = xmlData WHERE id = examFileId;
    END IF;
END;
$BODY$
  LANGUAGE plpgsql VOLATILE
  COST 100;

很高兴你找到了解决办法。老实说,由于与语言的层次结构相关的问题,字符串格式化函数在SGML中很难可靠地使用。也就是说,regexp对其功能有严格的限制

更好的解决方案可能是采用完全不同的方向,用PL/PerlU或PL/Python编写函数,并使用这些语言的现有XML处理功能。这可能会为您提供更好、更健壮的解决方案