Sql sp_send_dbmail attach文件作为varbinary存储在数据库中

Sql sp_send_dbmail attach文件作为varbinary存储在数据库中,sql,sp-send-dbmail,Sql,Sp Send Dbmail,我有一个两部分的问题,关于使用sp_send_dbmail将查询结果作为附件发送 问题1:仅打开基本的.txt文件。任何其他格式(如.pdf或.jpg)都已损坏 问题2:尝试发送多个附件时,我收到一个文件,所有文件名都粘在一起 我正在运行SQL Server 2005,我有一个存储上载文档的表: CREATE TABLE [dbo].[EmailAttachment]( [EmailAttachmentID] [int] IDENTITY(1,1) NOT NULL, [MassEmailID]

我有一个两部分的问题,关于使用sp_send_dbmail将查询结果作为附件发送

问题1:仅打开基本的.txt文件。任何其他格式(如.pdf或.jpg)都已损坏

问题2:尝试发送多个附件时,我收到一个文件,所有文件名都粘在一起

我正在运行SQL Server 2005,我有一个存储上载文档的表:

CREATE TABLE [dbo].[EmailAttachment](
[EmailAttachmentID] [int] IDENTITY(1,1) NOT NULL,
[MassEmailID] [int] NULL, -- foreign key
[FileData] [varbinary](max) NOT NULL,
[FileName] [varchar](100) NOT NULL,
[MimeType] [varchar](100) NOT NULL
我也有一个标准邮件的MassEmail表。下面是SQL发送邮件脚本。为简洁起见,我排除了declare语句

while ( (select count(MassEmailID) from MassEmail where status = 20 )>0) 
begin
    select @MassEmailID = Min(MassEmailID) from MassEmail where status = 20
    select @Subject = [Subject] from MassEmail where MassEmailID = @MassEmailID
    select @Body = Body from MassEmail where MassEmailID = @MassEmailID

    set @query = 'set nocount on; select cast(FileData as varchar(max)) from Mydatabase.dbo.EmailAttachment where MassEmailID = '+ CAST(@MassEmailID as varchar(100))  

    select  @filename = ''
    select  @filename = COALESCE(@filename+ ',', '') +FileName from EmailAttachment where MassEmailID = @MassEmailID

exec msdb.dbo.sp_send_dbmail    
    @profile_name = 'MASS_EMAIL',
    @recipients = 'me@myemail.com',
    @subject = @Subject,
    @body =@Body,
    @body_format ='HTML',
    @query = @query,
    @query_attachment_filename = @filename,
    @attach_query_result_as_file = 1,
    @query_result_separator = '; ',
    @query_no_truncate = 1,
    @query_result_header = 0;

update MassEmailset status= 30,SendDate = GetDate() where MassEmailID = @MassEmailID
end   
我能够成功地从数据库中读取文件,因此我知道二进制数据没有损坏

.txt文件仅在我将FilaData强制转换为varchar时读取。但很明显,原始标题丢失了。还值得注意的是,附件文件的大小与原始文件不同。这很可能也是由于编码不当造成的。所以我希望有一种方法可以使用存储的mimetype创建文件头,或者在二进制数据中包含文件头


我对最后几个参数的值也没有信心,我知道coalesce不太正确,因为它在第一个文件名前面加了一个逗号。但好的文档几乎不可能找到。请帮忙

我认为您无法直接从SQL发送二进制数据。有很多人在谈论同样的问题。从返回到查询的文本被格式化为文本文件。二进制被格式化为十六进制。正如您所指出的,它会损坏任何非文本文档的文件

我认为您仍然可以完成您试图做的事情,但是首先将二进制数据导出到文件系统,然后通过sendmail提供的传统文件附件方法将其导入

像这样。(仅概念-未经测试的代码)


谢谢你的回复。看起来它可以工作,但安全限制禁止运行xp\u cmdshell,尤其是用户上传的文件。看来我得换一种方式了。在这个问题上运气好吗。
DECLARE @OutputFileAndPath VarChar(500) = '\\Log_Files\MyFile.pdf ' 
DECLARE @sql VarChar(8000)

SELECT @sql = 'BCP "SELECT MyFile FROM [dbo].[MyTable] 
    WHERE PrimaryKey = 12345" queryout ' + @OutputFileAndPath +
        ' -S MyServer\MyInstance -T -fC:\Documents.fmt'

/* you could use a generic format file that would cover most formats */

EXEC xp_cmdshell @sql, NO_OUTPUT;

while ( (select count(MassEmailID) from MassEmail where status = 20 )>0) 
begin
    select @MassEmailID = Min(MassEmailID) from MassEmail where status = 20
    select @Subject = [Subject] from MassEmail where MassEmailID = @MassEmailID
    select @Body = Body from MassEmail where MassEmailID = @MassEmailID


    exec msdb.dbo.sp_send_dbmail    
        @profile_name = 'MASS_EMAIL',
        @recipients = 'me@myemail.com',
        @subject = @Subject,
        @body =@Body,
        @body_format ='HTML',
        @file_attachments = @OutputFileAndPath /* i.e. \\Log_Files\MyFile.pdf */

    update MassEmailset status= 30,SendDate = GetDate() where MassEmailID = @MassEmailID
end