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

ENH: Add support for executing UDF's using Bodo as the engine #60668

Open
1 of 3 tasks
scott-routledge2 opened this issue Jan 6, 2025 · 2 comments · May be fixed by #60622
Open
1 of 3 tasks

ENH: Add support for executing UDF's using Bodo as the engine #60668

scott-routledge2 opened this issue Jan 6, 2025 · 2 comments · May be fixed by #60622
Labels
Enhancement Needs Triage Issue that has not been reviewed by a pandas team member

Comments

@scott-routledge2
Copy link

Feature Type

  • Adding new functionality to pandas

  • Changing existing functionality in pandas

  • Removing existing functionality in pandas

Problem Description

Applying User Defined Functions (UDFs) to a DataFrame can be very slow when evaluated using the default python engine. Passing engine="numba" and leveraging Numba's Just-in-Time (JIT) compiler to transform the UDF application into an optimized binary can improve performance, however there are several limitations to the Numba UDF engine including:

  • Limited set of dtypes supported (only supports numpy dtypes, does not support ExtensionDtypes)
  • Parallel execution not supported (unless raw=True)
  • Difficulty troubleshooting issues due to lengthy stack traces and hard-to-read error messages.

Adding support for the Bodo engine would solve the above issues and provide a good complement to the capabilities of the currently supported engines (Python and Numba).

Bodo uses an auto-parallelizing JIT compiler to transform Python code into highly optimized, parallel binaries with an MPI backend, allowing it to scale to very large data sizes with minimal extra work required from the user (large speedups on both laptops and clusters). Bodo is also built for Pandas and supports DataFrame, Series and Array Extension types natively.

Feature Description

Allow passing the value "bodo" to the engine parameter in DataFrame.apply and add an apply_bodo method which accepts the user defined function and creates a jit function to do the apply and calls it. For example:
In pandas/core/apply.py

class FrameApply(NDFrameApply):
...
    def apply_series_bodo(self) -> DataFrame | Series:
        bodo = import_optional_dependency("bodo")

        engine_kwargs = bodo_get_jit_arguments(self.engine_kwargs)

        @bodo.jit(**engine_kwargs)
        def do_apply(obj, func, axis):
            return obj.apply(func, axis)

        result = do_apply(self.obj, self.func, self.axis)
        return result

This approach could also be applied to other API's that accepts a UDF and engine argument.

Alternative Solutions

Users could execute their UDF using a Bodo JIT'd function. For example:

import bodo
import pandas as pd

def f(x):
  return x.A // x.B if x.B != 0 else 0

@bodo.jit
def apply_udf(df, func):
  return df.apply(func, axis=1)

df = pd.DataFrame({"A": [1,2,3,4,5], "B": [0, 1, 2, 2, 2]})

result = apply_udf(df, f)

While this approach is fine, it has it's downsides such as requiring a larger code rewrite which could make it more difficult to quickly experiment with different engines.

Additional Context

Relevant links:
Bodo's documentation
Bodo's github repo
Proof-of-concept PR that adds support for engine="bodo" in df.apply.

@scott-routledge2 scott-routledge2 added Enhancement Needs Triage Issue that has not been reviewed by a pandas team member labels Jan 6, 2025
@scott-routledge2 scott-routledge2 linked a pull request Jan 6, 2025 that will close this issue
5 tasks
@jbrockmendel
Copy link
Member

jbrockmendel commented Jan 7, 2025

I don’t see why this needs to live in pandas instead of something like bodo.apply(func, df). Same amount of changed code for users.

@datapythonista
Copy link
Member

@jbrockmendel while in general I agree with your point, we already have the numba engine in pandas. Do you think we should remove it?

For what I understand, seems like Bodo should be better for most users, as it works with Arrow types (besides the other advantages discussed). So, while I'm a big fan of not adding more things into pandas, I don't see why we should have numba and not bodo. Is there a reason?

Also, assuming Bodo is included, are there reasons to keep numba? When is numba a better choice than Bodo?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Enhancement Needs Triage Issue that has not been reviewed by a pandas team member
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants