Using closedXML C# library, How can I figure out what data is causing this error when trying to save

前端 未结 3 1003
逝去的感伤
逝去的感伤 2021-01-18 11:17

I have a C# app that exports to Excel using ClosedXML. It works fine but just ran into an issue where when i hit the :

  var ms = new MemoryStream();
  work         


        
3条回答
  •  伪装坚强ぢ
    2021-01-18 11:32

    Since ClosedXML doesn't prevent you from using the 0x0B character in values, you'll either have to scrub your data of it yourself (as suggested by @Raidri), or you could force and exception, or do a string replace when the value is set. I've created a sample program below which uses Castle's Dynamic Proxy to wrap the IXLWorksheet and IXLCell interfaces. Firstly, we proxy the IXLWorksheet values (which returned from adding a new worksheet as in the example below, or by indexing an existing worksheet). This needs to be done manually via a method call; everything else from then on is set up. When accessing cells (via the Cell methods, or the ActiveCell property) a proxied IXLCell value is returned which checks the data being set via the Value property and the SetValue method. The check is done in the ValidateMethodInterceptor as per the comments. This whole mechanism can be left in your codebase and turned on/off via a switch in the Program.Proxy method if you so desire.

    As a further alternative, the package EPPlus (which has similar functionality to ClosedXML) doesn't crash when confronted with the VT character. Instead it replaces it with the value _x00B_. Perhaps a switch would be more beneficial?

    internal class Program
    {
        private static void Main(string[] args)
        {
            var stream = new MemoryStream();
    
            using (stream)
            {
                using (var workbook = new XLWorkbook())
                {
                    using (var worksheet = Proxy(workbook.Worksheets.Add("Sheet 1")))
                    {
                        worksheet.Cell("A1").Value = "This  is a test";
                        worksheet.Cell("A2").Value = "This \v is a test";
                        workbook.SaveAs(stream);
                    }
                }
            }
        }
    
        public static IXLWorksheet Proxy(IXLWorksheet target)
        {
            var generator = new ProxyGenerator();
            var options = new ProxyGenerationOptions { Selector = new WorksheetInterceptorSelector() };
    
            return generator.CreateInterfaceProxyWithTarget(target, options);
        }
    }
    
    public class WorksheetInterceptorSelector : IInterceptorSelector
    {
        private static readonly MethodInfo[] methodsToAdjust;
        private readonly ProxyCellInterceptor proxyCellInterceptor = new ProxyCellInterceptor();
    
        static WorksheetInterceptorSelector()
        {
            methodsToAdjust = typeof(IXLWorksheet).GetMethods()
                .Where(x => x.Name == "Cell")
                .Union(new[] { typeof(IXLWorksheet).GetProperty("ActiveCell").GetGetMethod() })
                .ToArray();
        }
    
        #region IInterceptorSelector Members
    
        public IInterceptor[] SelectInterceptors(System.Type type, System.Reflection.MethodInfo method, IInterceptor[] interceptors)
        {
            if (!methodsToAdjust.Contains(method))
                return interceptors;
    
            return new IInterceptor[] { proxyCellInterceptor }.Union(interceptors).ToArray();
        }
    
        #endregion
    }
    
    public class CellInterceptorSelector : IInterceptorSelector
    {
        private static readonly MethodInfo[] methodsToAdjust = new[] { typeof(IXLCell).GetMethod("SetValue"), typeof(IXLCell).GetProperty("Value").GetSetMethod() };
    
        private ValidateMethodInterceptor proxyCellInterceptor = new ValidateMethodInterceptor();
    
        #region IInterceptorSelector Members
    
        public IInterceptor[] SelectInterceptors(System.Type type, MethodInfo method, IInterceptor[] interceptors)
        {
            if (method.IsGenericMethod && method.Name == "SetValue" || methodsToAdjust.Contains(method))
                return new IInterceptor[] { proxyCellInterceptor }.Union(interceptors).ToArray();
            return interceptors;
        }
    
        #endregion
    }
    
    public class ProxyCellInterceptor : IInterceptor
    {
        #region IInterceptor Members
    
        public void Intercept(IInvocation invocation)
        {
            invocation.Proceed();
    
            //Wrap the return value
            invocation.ReturnValue = Proxy((IXLCell)invocation.ReturnValue);
        }
    
        #endregion
    
        public IXLCell Proxy(IXLCell target)
        {
            var generator = new ProxyGenerator();
            var options = new ProxyGenerationOptions { Selector = new CellInterceptorSelector() };
    
            return generator.CreateInterfaceProxyWithTarget(target, options);
        }
    }
    
    public class ValidateMethodInterceptor : IInterceptor
    {
        #region IInterceptor Members
    
        public void Intercept(IInvocation invocation)
        {
            var value = invocation.Arguments[0];
    
            //Validate the data as it is being set
            if (value != null && value.ToString().Contains('\v'))
            {
                throw new ArgumentException("Value cannot contain vertical tabs!");
            }
    
            //Alternatively, you could do a string replace:
            //if (value != null && value.ToString().Contains('\v'))
            //{
            //    invocation.Arguments[0] = value.ToString().Replace("\v", Environment.NewLine);
            //}
    
            invocation.Proceed();
        }
    
        #endregion
    }
    

提交回复
热议问题