Search code examples
delphimemory-leaksdelphi-xe2datasnap

How to release DataSnap memory once connections are closed?


I have a TCP/IP DataSnap server running as a service [Session based LifeCycle] that continuously chews up memory and never comes back to the starting memory size even when there are no connections to it.

In order to eliminate My code as the culprit, I have modeled up a basic TCP/IP DataSnap server running as VCL [Session based LifeCycle] that serves a Server Method class [TDSServerModule] which only contains basic mathematical functions using native data types [no objects to create or free].

When I connect to said DataSnap server with a very thin client I get the same results. The Memory Usage continuously grows with each connection and sporadically grows when executing the server side methods from the client. Once the connections are closed the DataSnap Server never reduces its Memory Usage [even when left running without connections for 8 hrs].

Any suggestions as to why this occurs or more importantly how to curtail it?

I am using RAD Studio XE2 Update 4 HotFix 1.


Solution

  • Let me quote a "must read" article about DataSnap. This is about XE3 but i hope the code here would work for XE2 as well.

    Memory consumption

    One of the issues that I observed was related to memory consumption. Why Datasnap server consumes so much memory if the method called does absolutely nothing?

    Maybe I don’t know how to explain exactly but i will try. Basically the DataSnap creates a session for each HTTP connection that it receives. This session will be destroyed after 20 minutes, in other words, on the first 20 minutes of the test the memory consumption will only go up, after that it has the tendency of stabilize itself. I really have no idea why Datasnap does this. In a REST application I don’t see much sense in these sessions with a default configuration. Of course, sessions can be helpful, but i can’t understand why it’s a default configuration. Indeed, DataSnap doesn’t have a configuration for that. It appears like you just have to use this session control, without being able to choose otherwise (There is no documentation). The MORMot framework has a session control too but it’s configurable and doesn’t consumes so much memory.

    Anyway, there is a way around this problem. Daniele Teti wrote an article on his blog, take a look. The solution that I will show here was placed by him on his blog. Thanks Daniele.

    uses System.StrUtils, DataSnap.DSSession, Data.DBXPlatform;
    
    function TServerMethods1.HelloWorld: String;
     begin
    
     Result := 'Hello World';
     GetInvocationMetaData.CloseSession := True;
    end;
    

    After running this method the session will close and memory consumption will be lower. Of course still exists an overhead for creating and destroying this session.

    So it seems the best course for you is ending every server method with explicit memory cleanup, if that was possible in XE2. Then you'd better read thosee articles again and prepare for future scalability challenges.