Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Returning large table is slow and consumes lots of memory #60

Open
sambu16 opened this issue Mar 12, 2018 · 15 comments
Open

Returning large table is slow and consumes lots of memory #60

sambu16 opened this issue Mar 12, 2018 · 15 comments

Comments

@sambu16
Copy link

sambu16 commented Mar 12, 2018

We're calling an RFC that returns a table with 500,000 rows (120 fields) (KE24 report) and this is consuming>8Gb RAM on the server and fails because of this.

In an attempt to get around this I've implemented some callbacks so that rows can be processed directly to file. Whilst this does seem to work, it is still taking ~12 minutes to extract the 500,000 rows which isn't particularly fast. The batch_size in the code below doesn't seem to have any effect really, set to batches of 10,000 it takes ~8secs to build and ~3secs to process to a csv file.

My new code is below to potentially spark some ideas of how to get a large table of data out quickly. The bottleneck seems to be the processing of the data out of the resultset. Perhaps processing the fields for every row of the table is not the most efficient using WrapStructure?

Any ideas how to speed this up and keep the amount of used memory to a minimum would be greatly appreciated.

Thanks,
Pat

    def call_and_save_table(self, func_name, table_name, batch_size, fnOnStartTable, fnOnEndTable, fnOnGetTableRow, **params):
        """ Invokes a remote-enabled function module via RFC.

        :param func_name (str): Name of the function module that will be invoked.
        :param table_name (str): Name of the table to save
        :param batch_size (int): Number of records to pass into fnOnGetTableRow
        :param fnOnStartTable (func): Callback for starting the table
        :param fnOnEndTable (func): Callback for completing the table
        :param fnOnGetTableRow (func): Callback for processing rows
        :param params (kwargs): Parameter of the function module. All non optional
              IMPORT, CHANGING, and TABLE parameters must be provided.

        :return: Boolean

        :raises: :exc:`~pyrfc.RFCError` or a subclass
                 thereof if the RFC call fails.
        """
        cdef RFC_RC rc
        cdef RFC_ERROR_INFO errorInfo
        cdef unsigned paramCount

        cdef RFC_TABLE_HANDLE table_handle
        cdef RFC_TYPE_DESC_HANDLE table_type_desc
        cdef RFC_PARAMETER_DESC paramDesc
        cdef RFC_FIELD_DESC fieldDesc
        cdef unsigned i, rowCount, fieldCount

        funcName = fillString(func_name)
        if not self.alive:
            self._open()
        cdef RFC_FUNCTION_DESC_HANDLE funcDesc = RfcGetFunctionDesc(self._handle, funcName, &errorInfo)
        free(funcName)
        if not funcDesc:
            self._error(&errorInfo)
        cdef RFC_FUNCTION_HANDLE funcCont = RfcCreateFunction(funcDesc, &errorInfo)
        if not funcCont:
            self._error(&errorInfo)
        try: # now we have a function module
            for name, value in params.iteritems():
                fillFunctionParameter(funcDesc, funcCont, name, value)
            with nogil:
                rc = RfcInvoke(self._handle, funcCont, &errorInfo)
            if rc != RFC_OK:
                self._error(&errorInfo)

            tableName = fillString(table_name)
            try:
                rc = RfcGetParameterDescByName(funcDesc, tableName, &paramDesc, &errorInfo)
                if rc != RFC_OK:
                    self._error(&errorInfo)
                table_type_desc = paramDesc.typeDescHandle
                rc = RfcGetTable(funcCont, tableName, &table_handle, &errorInfo)
                if rc != RFC_OK:
                    self._error(&errorInfo)
                writeTable(table_type_desc, table_handle, self._bconfig, batch_size, fnOnStartTable, fnOnEndTable, fnOnGetTableRow)
                return True
            finally:
                free(tableName)
        finally:
            RfcDestroyFunction(funcCont, NULL)

cdef writeTable(RFC_TYPE_DESC_HANDLE typeDesc, RFC_TABLE_HANDLE table_handle, config, batch_size, fnOnStartTable, fnOnEndTable, fnOnGetTableRow):
    cdef RFC_RC rc
    cdef RFC_ERROR_INFO errorInfo
    cdef unsigned i, fieldCount, rowCount
    # # For debugging in tables (cf. class TableCursor)
    # tc = TableCursor()
    # tc.typeDesc = typeDesc
    # tc.container = table_handle
    # return tc

    ## get the field names array and sort it, this can be passed to a DictWriter so the field order is predictable
    header_dict = {}
    headers_array = []
    field_desc_array = []
    cdef RFC_FIELD_DESC fieldDesc
    rc = RfcGetFieldCount(typeDesc, &fieldCount, &errorInfo)
    if rc != RFC_OK:
        raise wrapError(&errorInfo)

    for i in range(fieldCount):
        rc = RfcGetFieldDescByIndex(typeDesc, i, &fieldDesc, &errorInfo)
        if rc != RFC_OK:
            raise wrapError(&errorInfo)
        header_dict[wrapString(fieldDesc.name)] = wrapString(fieldDesc.name)
        field_desc_array[i] = fieldDesc

    headers_array = sorted(header_dict.keys())

    rc = RfcGetRowCount(table_handle, &rowCount, &errorInfo)
    if rc != RFC_OK:
        raise wrapError(&errorInfo)

    fnOnStartTable(headers_array, rowCount)

    batch = []
    for i in xrange(rowCount):
        rc = RfcMoveTo(table_handle, i, &errorInfo)
        if rc != RFC_OK:
            raise wrapError(&errorInfo)
        batch.append(wrapStructure(typeDesc, table_handle, config))
        if ((i+1) % batch_size == 0) or (i == rowCount-1):
            fnOnGetTableRow(batch)
            batch = []

    fnOnEndTable()

    return []
@bsrdjan
Copy link
Contributor

bsrdjan commented Mar 14, 2018

We're calling an RFC that returns a table with 500,000 rows (120 fields) (KE24 report) and this is consuming>8Gb RAM on the server and fails because of this.

Did you try making several RFC calls, returning less data, like 50 calls returning 10,000 rows each ?

Which RFC/BAPI is called and did you check how much of 12 min for all rows (or 8 sec for 10k rows) consumed on ABAP server and how much in Python?

Are all fields character and is the table passed as "Export" or "Tables" parameter type of the remote function module? "Tables" is faster.

@sambu16
Copy link
Author

sambu16 commented Mar 15, 2018

Did you try making several RFC calls, returning less data, like 50 calls returning 10,000 rows each ?

I will see if I can set this test up tomorrow.

Which RFC/BAPI is called and did you check how much of 12 min for all rows (or 8 sec for 10k rows) consumed on ABAP server and how much in Python?

It's a custom built RFC. It would seem like it spends about 2mins on ABAP Server before it starts working through the dataset on the python side, so 10 mins in Python - that's using my posted method above. With the original code, the ABAP time remains the same but it then the Python just consumes all the resources on the machine very quickly trying to fill the results.

Are all fields character and is the table passed as "Export" or "Tables" parameter type of the remote >
function module? "Tables" is faster.

There's a mixture of field types in the dataset. The table is passed as Tables parameter type.

@sambu16
Copy link
Author

sambu16 commented Mar 16, 2018

I ran the RFC daily to retrieve a full period (500,000 rows). The timing remained just short of 12mins.

Is it possible for pyRFC to retrieve the field list at the start of a table and pass that info to process each row rather than retrieving the info for every row in the table? That may speed up processing a little.

As for the memory consumption, I'm not sure why pyRFC uses so much to build the results. Consuming > 8Gb for 500,000 records seems way too much. It's 305Mb in a CSV file.

@guettli
Copy link
Contributor

guettli commented Apr 6, 2018

@bsrdjan you said "Did you try making several RFC" could you please elaborate what you mean?

Imagine I have 10 methods which all could return a lot of rows. Do I need to implement the chunking 10 times?

@bsrdjan
Copy link
Contributor

bsrdjan commented Apr 17, 2018

Yes, I meant chunking 10 times , which would be in this scenario inconvenient.

I am still thinking how to replicate and investigate the issue.

Which RFC is called? Is it perhaps standard RFC, I could test in my test system as well?

@patbuxton, regarding field type descriptions' caching, it would not bring any visible improvement. Descriptions are already internally cached in SAP NW RFC library, like function/parameter descriptions.

@sambu16
Copy link
Author

sambu16 commented Apr 18, 2018

I'll have to find out about the RFC for you, I believe the RFC was custom built but I'll have to check. Maybe you can replicate the memory consumption issue by a quick hack to loop 500,000 time retrieving the same table row?

@bsrdjan
Copy link
Contributor

bsrdjan commented Apr 18, 2018

hm, good idea, will try that 👍 If custom, could you eventually have a quick look inside, which BAPI or function modules does the most of the work? Then I could wrap that or these ones and try to replicate rows.

As a side note, I published a new release yesterday, offering a new feature to skip returning parameters, like large tables, which are not needed on client side.

Names of this parameters can be provided for the call() method and they will not be "requested" (supplied) on ABAP side. It can also help further reduce memory consumption.

@sambu16
Copy link
Author

sambu16 commented Jul 19, 2018

@bsrdjan, sorry for the delay getting back to you on this. The RFC we called was custom, I believe that BAPI_COPAQUERY_GETCOST_ACTDATA should do something similar. Did you ever try a loop to check out the memory consumption issue?

@bsrdjan
Copy link
Contributor

bsrdjan commented Aug 8, 2018

Hi @patbuxton and sorry it took so long for my reply.

I reviewed wrapStructure and other PyRFC code-sections and ran memory_profiler while using STFC_PERFORMANCE function module, with following parameters:

COUNT=99999
result = conn.call('STFC_PERFORMANCE',  
    **{'CHECKTAB': 'X', 'LGET0332': str(COUNT), 'LGET1000': str(COUNT)})

I could not find anything to improve in PyRFC, to reduce memory consumption.

After (big) ABAP tables are retrieved, they are stored in SAP NW RFC lib cache and by wrapResult/Table/Structure transformed into Python data. After that done, the cache marked free, by RfcDestroyFunction call.

The design is not optimal for large tables and processing in batches/chunks, as you already implemented, would be a nice feature. Adding it in standard requires more analysis, work and testing, including the RfcDestroyFunction postpone, after all chunks processed. Unless more customers request this feature, the feature is not a priority atm.

I also tried adding RfcDeleteCurrentRow, to free rows immediately after added to chunk but it kills the performance when 100k or more records used.

As workaround for the time being, could BAPI_COPAQUERY_GETCOST_ACTDATA or another RFM called on ABAP side, be extended with some sort of range selection parameters, so that it can be called multiple times, returning chunks of ABAP data?

@guettli
Copy link
Contributor

guettli commented Aug 14, 2018

I have been thinking about this again. I do not have large input/output tables at the moment, but maybe in the future.

It would be nice to have a documented work-around.

Would it make sense to store the large result in a content-server and return the content-ID to the client?

@bsrdjan
Copy link
Contributor

bsrdjan commented Aug 14, 2018

Yes, the content-server could be an option, although also not the simple one.

The solution / workaround depends also on use-case I think.

BAPIs and remote Function Modules (RFM) usually offer input parameters to narrow/select the amount of data records to be returned. The Python client could simply execute multiple calls, with different selection parameters, processing results as usual. Chunks would be in this case done already on backend/ABAP side, minimising the network traffic and memory required in Python. If the standard BAPI/RFM does not provide selection criteria, it should not a big effort to add selection capabilities in custom RFM wrapper, depends on specific case of course. This looks like the simplest approach to me.

PyHDB could eventually also be an option, for reading raw HANA tables' data, bypassing RFMs and ABAP post-processing? The selection should not be an issue here but the data-model knowledge, down to the filed level is required, which might be an issue. Also no existing ABAP business logic would be applied to filter/consolidate data.

I will share the question with more experts, perhaps more ideas come up.

@bsrdjan
Copy link
Contributor

bsrdjan commented Aug 16, 2018

I changed the title to "Chunking big data", to sync with related stackoverflow question: https://stackoverflow.com/questions/49691298/sap-rfc-chunking-of-big-data/51862805#51862805

I will also check possible optimisation, to delete each table row from C heap (SAP NW RFC SDK), after "wrapping" it in Python. If works, it would reduce memory consumption for tables, by not keeping C and Python full table copies in parallel.

@sambu16
Copy link
Author

sambu16 commented Aug 16, 2018

Just for reference, the memory issue does not occur when I use the change detailed in my first post to stream the values directly to a file. The memory issue would then seem to be related to building the result dicts in Python which my method avoids. Thanks for the continued investigation!

@bsrdjan
Copy link
Contributor

bsrdjan commented Aug 20, 2018

Just published the 1.9,93 release, with tables' processing optimised as follows.

The wrapTable method (reading ABAP tables) had two copies of data, exactly as described in stackoverflow comment. The new logic now "pipes" table rows from C/C++ NW RFC SDK to Python, deleting each row in C/C++, after re-created in Python. Two full copies of data, Python and C++, do not exist anymore at the same time and the memory consumption should be reduced to half, when reading ABAP tables.

The fillTable method is refactored and can be easily modified, if the same behaviour required when sending big tables from Python to ABAP. To cut the consumption here to half, the lines 1646 and 1654:

line = lines[i]
i += 1

need to be replaced respectively by:

line = lines[0]
del lines[:1]

After this change is implemented, each Python table row is deleted after passing to C/C++ NW RFC SDK, which saves the memory but also makes Python input tables "gone", after the RFC call is executed. It would be inconvenient for other use-cases and I left fillTables therefore just ready for the simple modification, to activate this feature.

Another option would be adding the list of table names to be processed this way, as mew RFC call option, which further complicates the interface and testing.

It would be great if you could share some figures about the consumption before/after this change?

This should keep the amount of used memory to minimum (with the current pyrfc design). If that not enough, the next step could be sending rows to external server for processing, similar to @patbuxton solution.

Regarding pagination/batches/chunks, for the time being the only option is at Python application level because PyRFC rework would require considerably more efforts.

@bsrdjan
Copy link
Contributor

bsrdjan commented Jun 18, 2019

Was the answer of any help? Shall we close the issue?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

3 participants