We have a large process in our application that runs once a month. This process typically runs in about 30 minutes and generates 342000 or so log events. Recently we updated our logging to a centralized model using WCF and are now having difficulty with performance. Whereas the previous solution would complete in about 30 minutes, with the new logging, it now takes 3 or 4 hours. The problem it seems is because the application is actually waiting for the WCF request to complete before execution continues. The WCF method is already configured as IsOneWay and I wrapped the call on the client side to that WCF method in a different thread to try to prevent this type of problem but it doesn’t seem to have worked. I have thought about using the async WCF calls but thought before I tried something else I would ask here to see if there is a better way to handle this.
Share
342000 log events in 30 minutes, if I did my math correctly, comes out to 190 log events per second. I think your problem may have to do with the default throttling settings in WCF. Even if your method is set to one-way, depending on if you’re creating a new proxy for each logged event, calling the method will still block while the proxy is created, the channel is opened, and if you’re using an HTTP-based binding, it will block until the message has been received by the service (an HTTP-based binding sends back a null response for a 1-way method call when the message is received). The default WCF throttling limits concurrent instances to 10 on the service side, which means only 10 requests will be handled at a time, and any further requests will get queued, so pair that with an HTTP binding, and anything after the first 10 requests are going to block at the client until it’s one of the 10 requests getting handled. Without knowing how your services are configured (instance mode, etc.) it’s hard to say more than that, but if you’re using per-call instancing, I’d recommend setting
MaxConcurrentCallsandMaxConcurrentInstanceson yourServiceBehaviorto something much higher (the defaults are 16 and 10, respectively).Also, to build on what others have mentioned about aggregating multiple events and submitting them all at once, I’ve found it helpful to setup a static
Logger.LogEvent(eventData)method. That way it’s simple to use throughout your code, and you can control in yourLogEventmethod how you want logging to behave throughout your application, such as configuring how many events should get submitted at a time.