草庐IT

xml - 将 XML 文件作为附件通过电子邮件发送时对内容传输编码感到困惑

coder 2024-06-25 原文

我有一个 UTF-8 编码的 XML 文件,它作为附件通过电子邮件发送。当电子邮件收件人打开电子邮件并保存附件时,XML 文件不再是 UTF-8(而是报告 ANSI 编码)。在这种情况下,收件人使用了 Microsoft Outlook(如果重要的话)。

我在无法依赖合适的 MIME 库的可用性的环境中进行编程,所以我需要了解我哪里出错了。

在通过电子邮件发送 XML 文件之前,在服务器上创建它之后,我可以使用 Linux file 命令看到它是一个 UTF-8 文件。除此之外,XML 还有一个版本头 <?xml version="1.0" encoding="UTF-8"?> (这与我的问题并不真正相关,但为了完整起见,我将其包括在内)。我很确定我通过电子邮件发送文件的代码是这里的问题,但我不确定执行此操作的“正确”方法。

我发送的标题是:

"Mime-Version" "1.0"
"Content-Type" "multipart/mixed; boundary="__==NAHDHDH2.28ABSDJxjhkjhsdkjhd___"\n\n"

电子邮件的正文是:
--__==NAHDHDH2.28ABSDJxjhkjhsdkjhd___\n
Content-Type: text/plain; charset="utf-8"; format=flowed\n
Content-Transfer-Encoding: 7bit\n\n
Please find attached the data file generated 
--__==NAHDHDH2.28ABSDJxjhkjhsdkjhd___\n
Content-Type: text/plain; charset="utf-8"\n
Content-Disposition: attachment; filename="My_File_Name"\n\n
XML FILE CONTENTS GO HERE
--__==NAHDHDH2.28ABSDJxjhkjhsdkjhd___--\n

问题:
  • 我应该使用 quoted-printable , 8bit或其他类型Content-Transfer-Encoding这里?我已经尝试了所有这些,但它
    没有改变结果。
  • Content-Type: text/plain XML 附件是否正确?
  • 还有其他建议吗?
  • 最佳答案

    通过指定 text/plain您基本上将控制权交给远程客户端的文本处理能力,这在这种特殊情况下显然是有限的。 XML 按照规范是 Unicode,因此通过选择更好的内容类型,您更有可能成功。试试 text/xmlapplication/xml相反,甚至是完全不透明的 application/octet-stream ,这应该只允许接收者以字节对字节相同的形式将其保存在磁盘上。
    内容传输编码根本不应该影响这种行为,但由于您似乎不清楚其重要性,这里是一个简短的讨论。
    内容传输编码是完全透明的;它不会影响交付的内容或远程客户端可以用它做什么。选择哪种内容传输编码取决于数据的性质和需要传输的电子邮件系统的功能。如果它不是 8 位干净的,则需要一个 7 位 CTE 将其封装成。如果内容的行太长而无法放入 SMTP,则需要将其封装为较短的行。但是远程客户端将提取另一端封装内的任何内容。在任何情况下使用。
    针对不同情况,内容传输编码具有层次结构:

  • 7bit如果您的数据完全是 7 位 ASCII 并且没有超过大约 990 个字符的行,则适用。然后它甚至可以在未经修改的情况下进行原始的旧 SMTP 传输。在没有任何明确的情况下 Content-Transfer-Encoding: header ,这是根据标准的默认值(尽管您经常看到其中包含 8 位数据的内容,而没有明确的 CTE,甚至没有明确的 7bit 声明)。
  • 8bit放宽了对数据 7 位清洁的要求。如果传输此消息的所有系统都支持 ESMTP 8BITMIME扩展,这对于行长受限的数据应该没问题。
  • binary另外允许无限的线长。理论上,您应该能够使用它来传递不受限制的内容,但实际上,当系统不严格遵守规范时,这似乎会引发故障。一个典型的症状是过长的线路在运输过程中被截断或折叠,破坏了有效载荷的完整性。为了避免此类问题(并更好地遵守互操作性标准的文字和精神),您最好采用以下方法之一。
  • base64接受不受限制的内容,但以一种满足限制行长度和严格限制的 7 位字符库的严格要求的格式对其进行编码。它将有效载荷扩展到原始大小的 4/3 以上。例子:
  •     ugqcA7R5cPq667vNaSifRUH9HsW00NqZ1gwICk0pNrUkXFpNIFOpbf3o
        5ml8cqqSygkp8KBgPbHrqnDXvZTEBOkNo7ThE+BAvexa75Tm0Ebo/Yjl
        y697pMp1+dnSlk3YTqxkPI9vqpple13dXLHlvnFDmSi0gqIMSwo7kUFD
        SivAWhyCBR6tFO3lY1Pk6lz78+zgL28VthI72kVRkrWWtzoFef/4u5Ip
        GR00CtsNNEJo01GAQGpkTNFT9U9Q/UI9CMGgaI9E9RkMaTDTQICBEyaE
        woSCQOrNGA==
    
  • quoted-printable类似地接受任意内容,但将所选字节编码为原始内容的 3 倍。当大部分输入是 ASCII 时,这是一个可以容忍的开销。换句话说,这适用于偶尔有非 ASCII 内容的粗略文本格式,例如使用 8 位编码的许多西方语言中的文本,或者像 HTML 这样的格式,其中 ASCII 标记在实际内容上占主导地位,几乎在任何情况下语。示例:
  •     <?xml version=3D"1.0" encoding=3D"UTF-8"?>h=C3=ABll=C3=B6 =
        w=C3=B6rld
    
    引用 printable 根本不难实现,并且似乎适合您的场景。
    所有这些都编入 MIME RFC 2045到 2048 年。维基百科有很好的可读文章,例如base64quoted-printable .
    从您的描述中不清楚您是刚刚将内容声明为可引用打印,还是对其进行了实际编码。我见过人们做前者,当它不起作用时表现出惊讶,但希望你做了后者。只是一个警示故事。

    关于xml - 将 XML 文件作为附件通过电子邮件发送时对内容传输编码感到困惑,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34999233/

    有关xml - 将 XML 文件作为附件通过电子邮件发送时对内容传输编码感到困惑的更多相关文章

    1. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

      我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

    2. ruby - 其他文件中的 Rake 任务 - 2

      我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时

    3. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

      我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

    4. ruby-on-rails - Rails 3 中的多个路由文件 - 2

      Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

    5. ruby - 将差异补丁应用于字符串/文件 - 2

      对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

    6. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

      我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

    7. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

      使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

    8. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

      我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

    9. ruby - RSpec - 使用测试替身作为 block 参数 - 2

      我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere

    10. Ruby 写入和读取对象到文件 - 2

      好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信

    随机推荐