gpt4 book ai didi

sql - sp_send_dbmail 附加在数据库中存储为 varbinary 的文件

转载 作者:行者123 更新时间:2023-12-01 19:35:05 28 4
gpt4 key购买 nike

我有一个由两部分组成的问题,涉及使用 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 发送邮件脚本。为了简洁起见,我排除了声明语句。

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 时读取。但显然原始 header 丢失了。还值得注意的是,附件文件的大小与原始文件的大小不同。这很可能也是由于编码不当造成的。所以我希望有一种方法可以使用存储的 mimetype 创建文件头,或者有某种方法可以在二进制数据中包含文件头?

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

最佳答案

我认为您无法直接从 SQL 发送二进制数据。有few posts out there谈论同样的问题。来自 Microsoft documentation查询返回的文本被格式化为文本文件。二进制格式为十六进制。正如您所指出的,这会损坏任何非文本文档的文件。

我认为您仍然可以首先完成您想要做的事情 using BCP将二进制数据导出到文件系统,然后通过 sendmail 可用的传统文件附件方法将其导回。

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

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

关于sql - sp_send_dbmail 附加在数据库中存储为 varbinary 的文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12431504/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com