EPPlus 2.9.0.1从SSIS包中保存大于~1.5 MiB的文件时抛出System.IO.IsolatedSto
本文关键字:文件 MiB IsolatedSto IO System SSIS 大于 保存 包中 EPPlus | 更新日期: 2023-09-27 18:09:13
问题
当我尝试用EPPlus保存超过1.5 MiB的文件时,ExcelPackage.Save()
抛出System.IO.IsolatedStorage.IsolatedStorageException
。
我正在用Visual Studio 2008 9.0.30729.4462 QFE
和.NET Framework 3.5 SP1
创建一个SSIS包,通过EPPlus 2.9.0.1
库导出SQL Server 2008 SP2 10.0.4311.0 64 bit
表的内容。
SSIS包真的很简单:一个Execute SQL Task
读取表的内容,并把它放在一个变量后面,一个Script task
读取recordset变量,并通过EPPlus将内容保存到磁盘。
脚本任务的代码为:
namespace ST_00a0b40814db4c7290b71f20a45b62c6.csproj
{
using System;
using System.AddIn;
using System.Data;
using System.Data.OleDb;
using System.IO;
using Microsoft.SqlServer.Dts.Runtime;
using Microsoft.SqlServer.Dts.Tasks.ScriptTask;
using OfficeOpenXml;
[AddIn("ScriptMain", Version = "1.0", Publisher = "", Description = "")]
public partial class ScriptMain : VSTARTScriptObjectModelBase
{
public void Main()
{
DataTable documentList = new DataTable();
using (OleDbDataAdapter adapter = new OleDbDataAdapter())
{
adapter.Fill(documentList, this.Dts.Variables["DocumentList"].Value);
}
if (documentList.Rows.Count > 0)
{
FileInfo fileInfo = new FileInfo(@"C:'Temp'Test.xlsx");
if (fileInfo.Exists)
{
fileInfo.Delete();
}
using (ExcelPackage package = new ExcelPackage(fileInfo))
{
ExcelWorksheet worksheet = package.Workbook.Worksheets.Add("Documents");
for (Int32 i = 0; i < documentList.Rows.Count; i++)
{
for (Int32 j = 0; j < documentList.Columns.Count; j++)
{
worksheet.Cells[i + 1, j + 1].Value = documentList.Rows[i][j];
}
}
package.Save();
}
}
Dts.TaskResult = Convert.ToInt32(DTSExecResult.Success);
}
}
}
当我只给脚本任务提供几个记录时,包运行得很好,但是当我对整个表运行它时,package.Save();
爆发了一个System.IO.IsolatedStorage.IsolatedStorageException: Unable to determine the identity of domain
异常。
这里可以看到完整的堆栈跟踪:
Error: System.Reflection.TargetInvocationException: Exception has been thrown by the target of an invocation. ---> System.InvalidOperationException: Error saving file C:'Temp'Test.xls ---> System.IO.IsolatedStorage.IsolatedStorageException: Unable to determine the identity of domain.
at System.IO.IsolatedStorage.IsolatedStorage._GetAccountingInfo(Evidence evidence, Type evidenceType, IsolatedStorageScope fAssmDomApp, Object& oNormalized)
at System.IO.IsolatedStorage.IsolatedStorage.GetAccountingInfo(Evidence evidence, Type evidenceType, IsolatedStorageScope fAssmDomApp, String& typeName, String& instanceName)
at System.IO.IsolatedStorage.IsolatedStorage._InitStore(IsolatedStorageScope scope, Evidence domainEv, Type domainEvidenceType, Evidence assemEv, Type assemblyEvidenceType, Evidence appEv, Type appEvidenceType)
at System.IO.IsolatedStorage.IsolatedStorage.InitStore(IsolatedStorageScope scope, Type domainEvidenceType, Type assemblyEvidenceType)
at System.IO.IsolatedStorage.IsolatedStorageFile.GetStore(IsolatedStorageScope scope, Type domainEvidenceType, Type assemblyEvidenceType)
at System.IO.IsolatedStorage.IsolatedStorageFile.GetUserStoreForDomain()
at MS.Internal.IO.Packaging.PackagingUtilities.ReliableIsolatedStorageFileFolder.GetCurrentStore()
at MS.Internal.IO.Packaging.PackagingUtilities.ReliableIsolatedStorageFileFolder..ctor()
at MS.Internal.IO.Packaging.PackagingUtilities.GetDefaultIsolatedStorageFile()
at MS.Internal.IO.Packaging.PackagingUtilities.CreateUserScopedIsolatedStorageFileStreamWithRandomName(Int32 retryCount, String& fileName)
at MS.Internal.IO.Packaging.SparseMemoryStream.EnsureIsolatedStoreStream()
at MS.Internal.IO.Packaging.SparseMemoryStream.SwitchModeIfNecessary()
at MS.Internal.IO.Packaging.SparseMemoryStream.Write(Byte[] buffer, Int32 offset, Int32 count)
at MS.Internal.IO.Packaging.CompressEmulationStream.Write(Byte[] buffer, Int32 offset, Int32 count)
at MS.Internal.IO.Packaging.CompressStream.Write(Byte[] buffer, Int32 offset, Int32 count)
at MS.Internal.IO.Zip.ProgressiveCrcCalculatingStream.Write(Byte[] buffer, Int32 offset, Int32 count)
at MS.Internal.IO.Zip.ZipIOModeEnforcingStream.Write(Byte[] buffer, Int32 offset, Int32 count)
at System.IO.StreamWriter.Flush(Boolean flushStream, Boolean flushEncoder)
at System.IO.StreamWriter.Write(String value)
at System.IO.TextWriter.Write(String format, Object arg0, Object arg1)
at OfficeOpenXml.ExcelWorksheet.UpdateRowCellData(StreamWriter sw)
at OfficeOpenXml.ExcelWorksheet.SaveXml()
at OfficeOpenXml.ExcelWorksheet.Save()
at OfficeOpenXml.ExcelWorkbook.Save()
at OfficeOpenXml.ExcelPackage.Save()
--- End of inner exception stack trace ---
at OfficeOpenXml.ExcelPackage.Save()
at ST_00a0b40814db4c7290b71f20a45b62c6.csproj.ScriptMain.Main() in C:'Temp'ScriptMain.cs:line 39
--- End of inner exception stack trace ---
at System.RuntimeMethodHandle._InvokeMethodFast(Object target, Object[] arguments, SignatureStruct& sig, MethodAttributes methodAttributes, RuntimeTypeHandle typeOwner)
at System.RuntimeMethodHandle.InvokeMethodFast(Object target, Object[] arguments, Signature sig, MethodAttributes methodAttributes, RuntimeTypeHandle typeOwner)
at System.Reflection.RuntimeMethodInfo.Invoke(Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture, Boolean skipVisibilityChecks)
at System.Reflection.RuntimeMethodInfo.Invoke(Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture)
at System.RuntimeType.InvokeMember(String name, BindingFlags bindingFlags, Binder binder, Object target, Object[] providedArgs, ParameterModifier[] modifiers, CultureInfo culture, String[] namedParams)
at System.Type.InvokeMember(String name, BindingFlags invokeAttr, Binder binder, Object target, Object[] args, CultureInfo culture)
at Microsoft.SqlServer.Dts.Tasks.ScriptTask.VSTATaskScriptingEngine.ExecuteScript()
我已经能够将问题定位到生成文件的大小:当Excel文件的大小增长约1.5 MiB时(这是一个或多或少的值,我无法找到确切的大小),错误出现。
我在网上能找到的唯一信息是一篇博客文章,博客作者提出了一个解决方案,他将代码"外包"到DLL并将其上传到服务器的GAC,然后运行以下代码:
AppDomainSetup setup = new AppDomainSetup();
setup.ApplicationBase = rootPath;
setup.DisallowBindingRedirects = false;
setup.DisallowCodeDownload = true;
setup.ConfigurationFile = AppDomain.CurrentDomain.SetupInformation.ConfigurationFile;
Evidence evidence = new Evidence();
evidence.AddHost(new Zone(SecurityZone.MyComputer));
AppDomain ad = AppDomain.CreateDomain("NewAppDomain", evidence, setup);
YourClass yourClass = (YourClass)ad.CreateInstanceAndUnwrap(typeof(YourClass).Assembly.FullName, typeof(YourClass).FullName);
yourClass.aMethod();
AppDomain.Unload(ad);
然而,我不能尝试这个解决方案,因为我没有访问服务器的GAC,我不能有DLL上传。
是否有其他方法可以绕过这个问题?
我还在EPPlus的问题跟踪器上打开了一个bug报告。
总结- SSIS包 Visual Studio 2008 9.0.30729.4462 QFE
- 。. NET Framework 3.5 SP
- SQL Server 2008 SP2 10.0.4311.0 64位
- EPPlus 2.9.0.1
- 大输出文件
- System.IO.IsolatedStorage。IsolatedStorageException:无法确定域 的身份
事实证明,您不需要创建一个全新的DLL并将其上传到GAC以使其工作。您可以在不同的AppDomain中创建原始类的新实例(或包含需要调用的方法的新类)。
1)将导致异常的代码放在一个单独的方法中。确保传入或传出的所有对象都是可序列化的或扩展MarshalByRefObject。在你的例子中:
public void SavePackage(FileInfo fileInfo, DataTable documentList)
{
using (ExcelPackage package = new ExcelPackage(fileInfo))
{
ExcelWorksheet worksheet = package.Workbook.Worksheets.Add("Documents");
for (Int32 i = 0; i < documentList.Rows.Count; i++)
{
for (Int32 j = 0; j < documentList.Columns.Count; j++)
{
worksheet.Cells[i + 1, j + 1].Value = documentList.Rows[i][j];
}
}
package.Save();
}
}
2)使用上面提到的代码创建类的新实例。
AppDomainSetup setup = new AppDomainSetup();
setup.ApplicationBase = AppDomain.CurrentDomain.BaseDirectory;
Evidence evidence = new Evidence(AppDomain.CurrentDomain.Evidence);
evidence.AddAssembly(Assembly.GetExecutingAssembly().FullName);
evidence.AddHost(new Zone(SecurityZone.MyComputer));
AppDomain ad = AppDomain.CreateDomain(DomainName, evidence, setup);
ScriptMain mainClass = (ScriptMain)ad.CreateInstanceAndUnwrap(typeof(ScriptMain).Assembly.FullName, typeof(ScriptMain).FullName);
3)调用方法,然后卸载AppDomain。
try
{
mainClass.SavePackage(fileInfo, documentList);
}
finally
{
AppDomain.Unload(ad);
}