← Back to team overview

yahoo-eng-team team mailing list archive

[Bug 1896125] Re: LDAP paging leaks memory

 

Reviewed:  https://review.opendev.org/754488
Committed: https://git.openstack.org/cgit/openstack/keystone/commit/?id=e98d1ac622f74cbbb41872226ab755bf4ff3ed84
Submitter: Zuul
Branch:    master

commit e98d1ac622f74cbbb41872226ab755bf4ff3ed84
Author: Lance Bragstad <lbragstad@xxxxxxxxx>
Date:   Tue Sep 29 10:20:13 2020 -0500

    Implement more robust connection handling for asynchronous LDAP calls
    
    Keystone's paging implementation contains a memory leak. The issue is
    noticeable if you integrate keystone with an LDAP server that supports
    paging and set `keystone.conf [ldap] page_size` to a low integer
    (e.g., 5).
    
    Keystone's LDAP backend uses `python-ldap` to interact with LDAP
    servers. For paged requests, it uses `search_ext()`, which is an
    asynchronous API [0]. The server responds with a message ID, which the
    client uses to retrieve all data for the request. In keystone's case,
    the `search_ext()` method is invoked with a page control that tells
    the server to deliver responses in increments according to the page
    size configured with `keystone.conf [ldap] page_size`. So long as the
    client has the original connection used to fetch the message ID, it
    can request the rest of the information associated to the request.
    
    Keystone's paging implementation loops continuously for paged
    requests. It takes the message ID it gets from `search_ext()` and
    calls `result3()`, asking the server for the data associated with that
    specific message. Keystone continues to do this until the server sends
    an indicator that it has no more data relevant to the query (via a
    cookie). The `search_ext()` and `result3()` methods must use the same
    LDAP connection.
    
    Given the above information, keystone uses context managers to provide
    connections. This is relevant when deploying connection pools, where
    certain connections are re-used from a pool. Keystone relies on Python
    context managers to handle connections, which is pretty typical
    use-case for context managers. Connection managers allow us to do the
    following (assuming pseudocode):
    
      with self.get_connection as conn:
          response = conn.search_s()
          return format(response)
    
    The above snippet assumes the `get_connection` method provides a
    connection object and a callable that implements `search_s`. Upon
    exiting the `with` statement, the connection is disconnected, or put
    back into the pool, or whatever the implementation of the context
    manager decides to do. Most connections in the LDAP backend are
    handled in this fashion.
    
    Unfortunately, the LDAP driver is somewhat oblivious to paging, it's
    control implementation, or the fact that it uses an asynchronous API.
    Instead, the driver leaves it up to the handler objects it uses for
    connections to determine if the request should be controlled via
    paging. This is an anti-pattern since the backend establishes the
    connection for the request but doesn't ensure that connection is
    safely handled for asynchronous APIs.
    
    This forces the `search_ext()` and `result3()` implementations in the
    PooledLDAPHandler to know how to handle connections and context
    managers, since it needs to ensure the same connection is used for
    paged requests. The current code tried to clean up the context
    manager responsible for connections after the results are collected
    from the server using the message ID. I believe it does this because
    it needs to get a new connection for each message in the paged
    results, even though it already operates from within a connection
    established via a context manager and the PooledLDAPHandler almost
    always returns the same connection object from the pool. The code
    tries to use a weak reference to create a callback that tears down the
    context manager when nothing else references it. At a high-level, the
    idea is similar to the following pseudocode:
    
      with self.get_connection as conn:
          while True:
            ldap_data = []
            context_manager = self.get_connection()
            connection = context_manager.__enter__()
            message_id = connection.search_ext()
            results = connection.result3(message_id)
            ldap_data.append(results)
            context_manager.__exit__()
    
    I wasn't able to see the callback get invoked or work as described in
    comments, resulting in memory bloat, especially with low page sizes
    which results in more requests. A weak reference invokes the callback
    when the weak reference is called, but there are no other references
    to the original object [1]. In our case, I don't think we invoke that
    path because we don't actually do anything with the weak reference. We
    assume it's going to run the callback when the object is garbage
    collected.
    
    This commit attempts to address this issue by using the concept of a
    finalizer [2], which was designed for similar cases. It also attempts
    to hide the cleanup implementation in the AsynchronousMessage object,
    so that callers don't have to worry about making sure they invoke the
    finalizer.
    
    An alternative approach would be to push more of the paging logic and
    implementation up into the LDAP driver. This would make it easier to
    put the entire asynchronous API flow for paging into a `with`
    statement and relying on the normal behavior of context managers to
    clean up accordingly. This approach would remove the manual cleanup
    invocation, regardless of using weak references or finalizer objects.
    However, this approach would likely require a non-trivial amount of
    design work to refactor the entire LDAP backend. The LDAP backend has
    other issues that would complicate the re-design process:
    
      - Handlers and connection are generalized to mean the same thing
      - Method names don't follow a convention
      - Domain-specific language from python-ldap bleeds into keystone's
        implementation (e.g., get_all, _ldap_get_all, add_member) at
        different points in the backend (e.g., UserApi (BaseLdap), GroupApi
        (BaseLdap), KeystoneLDAPHandler, PooledLDAPHandler,
        PythonLDAPHandler)
      - Backend contains dead code from when keystone supported writeable
        LDAP backends
      - Responsibility for connections and connection handling is spread
        across objects (BaseLdap, LDAPHandler)
      - Handlers will invoke methods differently based on configuration at
        runtime, which is a sign that the relationship between the driver,
        handlers, and connection objects isn't truely polymorphic
    
    While keeping the logic for properly handling context managers and
    connections in the Handlers might not be ideal, it is a relatively
    minimal fix in comparison to a re-design or backend refactor. These
    issues can be considered during a refactor of the LDAP backend if or
    when the community decides to re-design the LDAP backend.
    
    [0] https://www.python-ldap.org/en/python-ldap-3.3.0/reference/ldap.html#ldap.LDAPObject.search_ext
    [1] https://docs.python.org/3/library/weakref.html#weakref.ref
    [2] https://docs.python.org/3/library/weakref.html#finalizer-objects
    
    Closes-Bug: 1896125
    Change-Id: Ia45a45ff852d0d4e3a713dae07a46d4ff8d370f3


** Changed in: keystone
       Status: In Progress => Fix Released

-- 
You received this bug notification because you are a member of Yahoo!
Engineering Team, which is subscribed to OpenStack Identity (keystone).
https://bugs.launchpad.net/bugs/1896125

Title:
  LDAP paging leaks memory

Status in OpenStack Identity (keystone):
  Fix Released

Bug description:
  If you're using page_size [0] and are integrating keystone with an
  LDAP server that supports paging (like Active Directory), it's
  possible to see keystone memory footprint slowly increase over time.

  The problem isn't as noticable with large page sizes (e.g., page_size
  = 10000). But it's noticable when you use small page sizes (e.g.,
  page_size = 5).

  I hit this issue using Active Directory with 10,000 users. I set my
  page_size to 5 and listed users continuously for an hour. During that
  time I noticed keystone's total memory consumption on the host
  increase from 5% to 14%.

  Additionally, the problem is exacerbated using page_size = 1.

  I was unsuccessful in reproducing this issue with FreeIPA, which is
  another LDAP implementation, but it doesn't support paging. Keystone
  automatically disables paging if the LDAP server doesn't support it.

  It seems there is a memory leak somewhere in keystone's LDAP paging
  implementation.

  [0] https://docs.openstack.org/keystone/latest/configuration/config-
  options.html#ldap.page_size

To manage notifications about this bug go to:
https://bugs.launchpad.net/keystone/+bug/1896125/+subscriptions


References