From ff87c859d345e219a048ecf1190b64fc2abcd677 Mon Sep 17 00:00:00 2001 From: Tomer Weitzman <81749152+tomerbv@users.noreply.github.com> Date: Thu, 12 Feb 2026 11:40:03 +0200 Subject: [PATCH 1/3] Update scikit-learn to version 1.5 (#967) * updated scikit-learn~=1.5 fixes and patches for new scikit-learn version changes in item.yaml and regenerate function.yaml * remove filename * remove numpy import * revert sklearn.metrics monkey patch fix _get_dataframe to handle list/dict before accessing artifact_url added feature name preservation logic in predict function * revert mlrun version * revert get_or_create_project * revert scikit-learn version * scikit-learn==1.5.2 mlrun v 1.10 * scikit-learn==1.4.2 * revert scikit-learn<1.4.0 * scikit-learn~=1.5 * mlrun 1.10 with scikit-learn<1.4.0 * scikit-learn strict v~=1.5.2 added skip for test_train in test_auto_trainer.py * revert sklearn_classifier.py changes change XGBRegressor to LGBMRegressor * added xgboost.XGBRegressor, xgboost.XGBClassifier and lightgbm.LGBMClassifier models to test --- functions/src/auto_trainer/auto_trainer.py | 67 ++++++++------- functions/src/auto_trainer/function.yaml | 76 ++++++++--------- functions/src/auto_trainer/item.yaml | 4 +- functions/src/auto_trainer/requirements.txt | 3 +- .../src/auto_trainer/test_auto_trainer.py | 13 ++- functions/src/describe/function.yaml | 82 +++++++++---------- functions/src/describe/item.yaml | 4 +- functions/src/describe/requirements.txt | 2 +- functions/src/gen_class_data/function.yaml | 18 ++-- functions/src/gen_class_data/item.yaml | 4 +- functions/src/gen_class_data/requirements.txt | 2 +- .../src/gen_class_data/test_gen_class_data.py | 5 +- 12 files changed, 148 insertions(+), 132 deletions(-) diff --git a/functions/src/auto_trainer/auto_trainer.py b/functions/src/auto_trainer/auto_trainer.py index 7b4764700..4e53e5b7e 100755 --- a/functions/src/auto_trainer/auto_trainer.py +++ b/functions/src/auto_trainer/auto_trainer.py @@ -67,30 +67,14 @@ def _get_dataframe( Classification tasks. :param drop_columns: str/int or a list of strings/ints that represent the column names/indices to drop. """ - store_uri_prefix, _ = mlrun.datastore.parse_store_uri(dataset.artifact_url) - - # Getting the dataset: - if mlrun.utils.StorePrefix.FeatureVector == store_uri_prefix: - label_columns = label_columns or dataset.meta.status.label_column - context.logger.info(f"label columns: {label_columns}") - # FeatureVector case: - try: - fv = mlrun.datastore.get_store_resource(dataset.artifact_url) - dataset = fv.get_offline_features(drop_columns=drop_columns).to_dataframe() - except AttributeError: - # Leave here for backwards compatibility - dataset = fs.get_offline_features( - dataset.meta.uri, drop_columns=drop_columns - ).to_dataframe() - - elif not label_columns: - context.logger.info( - "label_columns not provided, mandatory when dataset is not a FeatureVector" - ) - raise ValueError - - elif isinstance(dataset, (list, dict)): + # Check if dataset is list/dict first (before trying to access artifact_url) + if isinstance(dataset, (list, dict)): # list/dict case: + if not label_columns: + context.logger.info( + "label_columns not provided, mandatory when dataset is not a FeatureVector" + ) + raise ValueError dataset = pd.DataFrame(dataset) # Checking if drop_columns provided by integer type: if drop_columns: @@ -103,17 +87,38 @@ def _get_dataframe( ) raise ValueError dataset.drop(drop_columns, axis=1, inplace=True) - else: - # simple URL case: - dataset = dataset.as_df() - if drop_columns: - if all(col in dataset for col in drop_columns): - dataset = dataset.drop(drop_columns, axis=1) - else: + # Dataset is a DataItem with artifact_url (URI or FeatureVector) + store_uri_prefix, _ = mlrun.datastore.parse_store_uri(dataset.artifact_url) + + # Getting the dataset: + if mlrun.utils.StorePrefix.FeatureVector == store_uri_prefix: + label_columns = label_columns or dataset.meta.status.label_column + context.logger.info(f"label columns: {label_columns}") + # FeatureVector case: + try: + fv = mlrun.datastore.get_store_resource(dataset.artifact_url) + dataset = fv.get_offline_features(drop_columns=drop_columns).to_dataframe() + except AttributeError: + # Leave here for backwards compatibility + dataset = fs.get_offline_features( + dataset.meta.uri, drop_columns=drop_columns + ).to_dataframe() + else: + # simple URL case: + if not label_columns: context.logger.info( - "not all of the columns to drop in the dataset, drop columns process skipped" + "label_columns not provided, mandatory when dataset is not a FeatureVector" ) + raise ValueError + dataset = dataset.as_df() + if drop_columns: + if all(col in dataset for col in drop_columns): + dataset = dataset.drop(drop_columns, axis=1) + else: + context.logger.info( + "not all of the columns to drop in the dataset, drop columns process skipped" + ) return dataset, label_columns diff --git a/functions/src/auto_trainer/function.yaml b/functions/src/auto_trainer/function.yaml index 0920b1033..50a36e750 100644 --- a/functions/src/auto_trainer/function.yaml +++ b/functions/src/auto_trainer/function.yaml @@ -1,22 +1,19 @@ -metadata: - categories: - - machine-learning - - model-training - tag: '' - name: auto-trainer +verbose: false +kind: job spec: - image: mlrun/mlrun - build: - origin_filename: '' - functionSourceCode: # Copyright 2019 Iguazio
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
from pathlib import Path
from typing import Any, Dict, List, Optional, Tuple, Union

import mlrun
import mlrun.datastore
import mlrun.utils
import pandas as pd
from mlrun import feature_store as fs
from mlrun.datastore import DataItem
from mlrun.execution import MLClientCtx
from mlrun.frameworks.auto_mlrun import AutoMLRun
from mlrun.utils.helpers import create_class, create_function
from sklearn.model_selection import train_test_split

PathType = Union[str, Path]


class KWArgsPrefixes:
    MODEL_CLASS = "CLASS_"
    FIT = "FIT_"
    TRAIN = "TRAIN_"


def _get_sub_dict_by_prefix(src: Dict, prefix_key: str) -> Dict[str, Any]:
    """
    Collect all the keys from the given dict that starts with the given prefix and creates a new dictionary with these
    keys.

    :param src:         The source dict to extract the values from.
    :param prefix_key:  Only keys with this prefix will be returned. The keys in the result dict will be without this
                        prefix.
    """
    return {
        key.replace(prefix_key, ""): val
        for key, val in src.items()
        if key.startswith(prefix_key)
    }


def _get_dataframe(
    context: MLClientCtx,
    dataset: DataItem,
    label_columns: Optional[Union[str, List[str]]] = None,
    drop_columns: Union[str, List[str], int, List[int]] = None,
) -> Tuple[pd.DataFrame, Optional[Union[str, List[str]]]]:
    """
    Getting the DataFrame of the dataset and drop the columns accordingly.

    :param context:         MLRun context.
    :param dataset:         The dataset to train the model on.
                            Can be either a list of lists, dict, URI or a FeatureVector.
    :param label_columns:   The target label(s) of the column(s) in the dataset. for Regression or
                            Classification tasks.
    :param drop_columns:    str/int or a list of strings/ints that represent the column names/indices to drop.
    """
    store_uri_prefix, _ = mlrun.datastore.parse_store_uri(dataset.artifact_url)

    # Getting the dataset:
    if mlrun.utils.StorePrefix.FeatureVector == store_uri_prefix:
        label_columns = label_columns or dataset.meta.status.label_column
        context.logger.info(f"label columns: {label_columns}")
        # FeatureVector case:
        try:
            fv = mlrun.datastore.get_store_resource(dataset.artifact_url)
            dataset = fv.get_offline_features(drop_columns=drop_columns).to_dataframe()
        except AttributeError:
            # Leave here for backwards compatibility
            dataset = fs.get_offline_features(
                dataset.meta.uri, drop_columns=drop_columns
            ).to_dataframe()

    elif not label_columns:
        context.logger.info(
            "label_columns not provided, mandatory when dataset is not a FeatureVector"
        )
        raise ValueError

    elif isinstance(dataset, (list, dict)):
        # list/dict case:
        dataset = pd.DataFrame(dataset)
        # Checking if drop_columns provided by integer type:
        if drop_columns:
            if isinstance(drop_columns, str) or (
                isinstance(drop_columns, list)
                and any(isinstance(col, str) for col in drop_columns)
            ):
                context.logger.error(
                    "drop_columns must be an integer/list of integers if not provided with a URI/FeatureVector dataset"
                )
                raise ValueError
            dataset.drop(drop_columns, axis=1, inplace=True)

    else:
        # simple URL case:
        dataset = dataset.as_df()
        if drop_columns:
            if all(col in dataset for col in drop_columns):
                dataset = dataset.drop(drop_columns, axis=1)
            else:
                context.logger.info(
                    "not all of the columns to drop in the dataset, drop columns process skipped"
                )

    return dataset, label_columns


def train(
    context: MLClientCtx,
    dataset: DataItem,
    model_class: str,
    label_columns: Optional[Union[str, List[str]]] = None,
    drop_columns: List[str] = None,
    model_name: str = "model",
    tag: str = "",
    sample_set: DataItem = None,
    test_set: DataItem = None,
    train_test_split_size: float = None,
    random_state: int = None,
    labels: dict = None,
    **kwargs,
):
    """
    Training a model with the given dataset.

    example::

        import mlrun
        project = mlrun.get_or_create_project("my-project")
        project.set_function("hub://auto_trainer", "train")
        trainer_run = project.run(
            name="train",
            handler="train",
            inputs={"dataset": "./path/to/dataset.csv"},
            params={
                "model_class": "sklearn.linear_model.LogisticRegression",
                "label_columns": "label",
                "drop_columns": "id",
                "model_name": "my-model",
                "tag": "v1.0.0",
                "sample_set": "./path/to/sample_set.csv",
                "test_set": "./path/to/test_set.csv",
                "CLASS_solver": "liblinear",
            },
        )

    :param context:                 MLRun context
    :param dataset:                 The dataset to train the model on. Can be either a URI or a FeatureVector
    :param model_class:             The class of the model, e.g. `sklearn.linear_model.LogisticRegression`
    :param label_columns:           The target label(s) of the column(s) in the dataset. for Regression or
                                    Classification tasks. Mandatory when dataset is not a FeatureVector.
    :param drop_columns:            str or a list of strings that represent the columns to drop
    :param model_name:              The model's name to use for storing the model artifact, default to 'model'
    :param tag:                     The model's tag to log with
    :param sample_set:              A sample set of inputs for the model for logging its stats along the model in favour
                                    of model monitoring. Can be either a URI or a FeatureVector
    :param test_set:                The test set to train the model with.
    :param train_test_split_size:   if test_set was provided then this argument is ignored.
                                    Should be between 0.0 and 1.0 and represent the proportion of the dataset to include
                                    in the test split. The size of the Training set is set to the complement of this
                                    value. Default = 0.2
    :param random_state:            Relevant only when using train_test_split_size.
                                    A random state seed to shuffle the data. For more information, see:
                                    https://scikit-learn.org/stable/glossary.html#term-random_state
                                    Notice that here we only pass integer values.
    :param labels:                  Labels to log with the model
    :param kwargs:                  Here you can pass keyword arguments with prefixes,
                                    that will be parsed and passed to the relevant function, by the following prefixes:
                                    - `CLASS_` - for the model class arguments
                                    - `FIT_` - for the `fit` function arguments
                                    - `TRAIN_` - for the `train` function (in xgb or lgbm train function - future)

    """
    # Validate inputs:
    # Check if exactly one of them is supplied:
    if test_set is None:
        if train_test_split_size is None:
            context.logger.info(
                "test_set or train_test_split_size are not provided, setting train_test_split_size to 0.2"
            )
            train_test_split_size = 0.2

    elif train_test_split_size:
        context.logger.info(
            "test_set provided, ignoring given train_test_split_size value"
        )
        train_test_split_size = None

    # Get DataFrame by URL or by FeatureVector:
    dataset, label_columns = _get_dataframe(
        context=context,
        dataset=dataset,
        label_columns=label_columns,
        drop_columns=drop_columns,
    )

    # Getting the sample set:
    if sample_set is None:
        context.logger.info(
            f"Sample set not given, using the whole training set as the sample set"
        )
        sample_set = dataset
    else:
        sample_set, _ = _get_dataframe(
            context=context,
            dataset=sample_set,
            label_columns=label_columns,
            drop_columns=drop_columns,
        )

    # Parsing kwargs:
    # TODO: Use in xgb or lgbm train function.
    train_kwargs = _get_sub_dict_by_prefix(src=kwargs, prefix_key=KWArgsPrefixes.TRAIN)
    fit_kwargs = _get_sub_dict_by_prefix(src=kwargs, prefix_key=KWArgsPrefixes.FIT)
    model_class_kwargs = _get_sub_dict_by_prefix(
        src=kwargs, prefix_key=KWArgsPrefixes.MODEL_CLASS
    )

    # Check if model or function:
    if hasattr(model_class, "train"):
        # TODO: Need to call: model(), afterwards to start the train function.
        # model = create_function(f"{model_class}.train")
        raise NotImplementedError
    else:
        # Creating model instance:
        model = create_class(model_class)(**model_class_kwargs)

    x = dataset.drop(label_columns, axis=1)
    y = dataset[label_columns]
    if train_test_split_size:
        x_train, x_test, y_train, y_test = train_test_split(
            x, y, test_size=train_test_split_size, random_state=random_state
        )
    else:
        x_train, y_train = x, y

        test_set = test_set.as_df()
        if drop_columns:
            test_set = dataset.drop(drop_columns, axis=1)

        x_test, y_test = test_set.drop(label_columns, axis=1), test_set[label_columns]

    AutoMLRun.apply_mlrun(
        model=model,
        model_name=model_name,
        context=context,
        tag=tag,
        sample_set=sample_set,
        y_columns=label_columns,
        test_set=test_set,
        x_test=x_test,
        y_test=y_test,
        artifacts=context.artifacts,
        labels=labels,
    )
    context.logger.info(f"training '{model_name}'")
    model.fit(x_train, y_train, **fit_kwargs)


def evaluate(
    context: MLClientCtx,
    model: str,
    dataset: mlrun.DataItem,
    drop_columns: List[str] = None,
    label_columns: Optional[Union[str, List[str]]] = None,
    **kwargs,
):
    """
    Evaluating a model. Artifacts generated by the MLHandler.

    :param context:                 MLRun context.
    :param model:                   The model Store path.
    :param dataset:                 The dataset to evaluate the model on. Can be either a URI or a FeatureVector.
    :param drop_columns:            str or a list of strings that represent the columns to drop.
    :param label_columns:           The target label(s) of the column(s) in the dataset. for Regression or
                                    Classification tasks. Mandatory when dataset is not a FeatureVector.
    :param kwargs:                  Here you can pass keyword arguments to the predict function
                                    (PREDICT_ prefix is not required).
    """
    # Get dataset by URL or by FeatureVector:
    dataset, label_columns = _get_dataframe(
        context=context,
        dataset=dataset,
        label_columns=label_columns,
        drop_columns=drop_columns,
    )

    # Parsing label_columns:
    parsed_label_columns = []
    if label_columns:
        label_columns = (
            label_columns if isinstance(label_columns, list) else [label_columns]
        )
        for lc in label_columns:
            if fs.common.feature_separator in lc:
                feature_set_name, label_name, alias = fs.common.parse_feature_string(lc)
                parsed_label_columns.append(alias or label_name)
        if parsed_label_columns:
            label_columns = parsed_label_columns

    x = dataset.drop(label_columns, axis=1)
    y = dataset[label_columns]

    # Loading the model and predicting:
    model_handler = AutoMLRun.load_model(
        model_path=model, context=context, model_name="model_LinearRegression"
    )
    AutoMLRun.apply_mlrun(model_handler.model, y_test=y, model_path=model)

    context.logger.info(f"evaluating '{model_handler.model_name}'")
    model_handler.model.predict(x, **kwargs)


def predict(
    context: MLClientCtx,
    model: str,
    dataset: mlrun.DataItem,
    drop_columns: Union[str, List[str], int, List[int]] = None,
    label_columns: Optional[Union[str, List[str]]] = None,
    result_set: Optional[str] = None,
    **kwargs,
):
    """
    Predicting dataset by a model.

    :param context:                 MLRun context.
    :param model:                   The model Store path.
    :param dataset:                 The dataset to predict the model on. Can be either a URI, a FeatureVector or a
                                    sample in a shape of a list/dict.
                                    When passing a sample, pass the dataset as a field in `params` instead of `inputs`.
    :param drop_columns:            str/int or a list of strings/ints that represent the column names/indices to drop.
                                    When the dataset is a list/dict this parameter should be represented by integers.
    :param label_columns:           The target label(s) of the column(s) in the dataset. for Regression or
                                    Classification tasks. Mandatory when dataset is not a FeatureVector.
    :param result_set:              The db key to set name of the prediction result and the filename.
                                    Default to 'prediction'.
    :param kwargs:                  Here you can pass keyword arguments to the predict function
                                    (PREDICT_ prefix is not required).
    """
    # Get dataset by URL or by FeatureVector:
    dataset, label_columns = _get_dataframe(
        context=context,
        dataset=dataset,
        label_columns=label_columns,
        drop_columns=drop_columns,
    )

    # loading the model, and getting the model handler:
    model_handler = AutoMLRun.load_model(model_path=model, context=context)

    # Dropping label columns if necessary:
    if not label_columns:
        label_columns = []
    elif isinstance(label_columns, str):
        label_columns = [label_columns]

    # Predicting:
    context.logger.info(f"making prediction by '{model_handler.model_name}'")
    y_pred = model_handler.model.predict(dataset, **kwargs)

    # Preparing and validating label columns for the dataframe of the prediction result:
    num_predicted = 1 if len(y_pred.shape) == 1 else y_pred.shape[1]

    if num_predicted > len(label_columns):
        if num_predicted == 1:
            label_columns = ["predicted labels"]
        else:
            label_columns.extend(
                [
                    f"predicted_label_{i + 1 + len(label_columns)}"
                    for i in range(num_predicted - len(label_columns))
                ]
            )
    elif num_predicted < len(label_columns):
        context.logger.error(
            f"number of predicted labels: {num_predicted} is smaller than number of label columns: {len(label_columns)}"
        )
        raise ValueError

    artifact_name = result_set or "prediction"
    labels_inside_df = set(label_columns) & set(dataset.columns.tolist())
    if labels_inside_df:
        context.logger.error(
            f"The labels: {labels_inside_df} are already existed in the dataframe"
        )
        raise ValueError
    pred_df = pd.concat([dataset, pd.DataFrame(y_pred, columns=label_columns)], axis=1)
    context.log_dataset(artifact_name, pred_df, db_key=result_set)
 - code_origin: '' - description: Automatic train, evaluate and predict functions for the ML frameworks - - Scikit-Learn, XGBoost and LightGBM. - disable_auto_mount: false - default_handler: train entry_points: train: - lineno: 121 + doc: "Training a model with the given dataset.\n\nexample::\n\n import mlrun\n\ + \ project = mlrun.get_or_create_project(\"my-project\")\n project.set_function(\"\ + hub://auto_trainer\", \"train\")\n trainer_run = project.run(\n \ + \ name=\"train\",\n handler=\"train\",\n inputs={\"dataset\"\ + : \"./path/to/dataset.csv\"},\n params={\n \"model_class\"\ + : \"sklearn.linear_model.LogisticRegression\",\n \"label_columns\"\ + : \"label\",\n \"drop_columns\": \"id\",\n \"model_name\"\ + : \"my-model\",\n \"tag\": \"v1.0.0\",\n \"sample_set\"\ + : \"./path/to/sample_set.csv\",\n \"test_set\": \"./path/to/test_set.csv\"\ + ,\n \"CLASS_solver\": \"liblinear\",\n },\n )" + has_kwargs: true parameters: - name: context type: MLClientCtx @@ -70,21 +67,12 @@ spec: type: dict doc: Labels to log with the model default: null - has_varargs: false + lineno: 126 name: train - has_kwargs: true - doc: "Training a model with the given dataset.\n\nexample::\n\n import mlrun\n\ - \ project = mlrun.get_or_create_project(\"my-project\")\n project.set_function(\"\ - hub://auto_trainer\", \"train\")\n trainer_run = project.run(\n \ - \ name=\"train\",\n handler=\"train\",\n inputs={\"dataset\"\ - : \"./path/to/dataset.csv\"},\n params={\n \"model_class\"\ - : \"sklearn.linear_model.LogisticRegression\",\n \"label_columns\"\ - : \"label\",\n \"drop_columns\": \"id\",\n \"model_name\"\ - : \"my-model\",\n \"tag\": \"v1.0.0\",\n \"sample_set\"\ - : \"./path/to/sample_set.csv\",\n \"test_set\": \"./path/to/test_set.csv\"\ - ,\n \"CLASS_solver\": \"liblinear\",\n },\n )" + has_varargs: false evaluate: - lineno: 273 + doc: Evaluating a model. Artifacts generated by the MLHandler. + has_kwargs: true parameters: - name: context type: MLClientCtx @@ -104,12 +92,12 @@ spec: doc: The target label(s) of the column(s) in the dataset. for Regression or Classification tasks. Mandatory when dataset is not a FeatureVector. default: null - has_varargs: false + lineno: 278 name: evaluate - has_kwargs: true - doc: Evaluating a model. Artifacts generated by the MLHandler. + has_varargs: false predict: - lineno: 327 + doc: Predicting dataset by a model. + has_kwargs: true parameters: - name: context type: MLClientCtx @@ -138,10 +126,22 @@ spec: doc: The db key to set name of the prediction result and the filename. Default to 'prediction'. default: null - has_varargs: false + lineno: 332 name: predict - has_kwargs: true - doc: Predicting dataset by a model. + has_varargs: false + build: + code_origin: '' + origin_filename: '' + functionSourceCode: # Copyright 2019 Iguazio
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
from pathlib import Path
from typing import Any, Dict, List, Optional, Tuple, Union

import mlrun
import mlrun.datastore
import mlrun.utils
import pandas as pd
from mlrun import feature_store as fs
from mlrun.datastore import DataItem
from mlrun.execution import MLClientCtx
from mlrun.frameworks.auto_mlrun import AutoMLRun
from mlrun.utils.helpers import create_class, create_function
from sklearn.model_selection import train_test_split

PathType = Union[str, Path]


class KWArgsPrefixes:
    MODEL_CLASS = "CLASS_"
    FIT = "FIT_"
    TRAIN = "TRAIN_"


def _get_sub_dict_by_prefix(src: Dict, prefix_key: str) -> Dict[str, Any]:
    """
    Collect all the keys from the given dict that starts with the given prefix and creates a new dictionary with these
    keys.

    :param src:         The source dict to extract the values from.
    :param prefix_key:  Only keys with this prefix will be returned. The keys in the result dict will be without this
                        prefix.
    """
    return {
        key.replace(prefix_key, ""): val
        for key, val in src.items()
        if key.startswith(prefix_key)
    }


def _get_dataframe(
    context: MLClientCtx,
    dataset: DataItem,
    label_columns: Optional[Union[str, List[str]]] = None,
    drop_columns: Union[str, List[str], int, List[int]] = None,
) -> Tuple[pd.DataFrame, Optional[Union[str, List[str]]]]:
    """
    Getting the DataFrame of the dataset and drop the columns accordingly.

    :param context:         MLRun context.
    :param dataset:         The dataset to train the model on.
                            Can be either a list of lists, dict, URI or a FeatureVector.
    :param label_columns:   The target label(s) of the column(s) in the dataset. for Regression or
                            Classification tasks.
    :param drop_columns:    str/int or a list of strings/ints that represent the column names/indices to drop.
    """
    # Check if dataset is list/dict first (before trying to access artifact_url)
    if isinstance(dataset, (list, dict)):
        # list/dict case:
        if not label_columns:
            context.logger.info(
                "label_columns not provided, mandatory when dataset is not a FeatureVector"
            )
            raise ValueError
        dataset = pd.DataFrame(dataset)
        # Checking if drop_columns provided by integer type:
        if drop_columns:
            if isinstance(drop_columns, str) or (
                isinstance(drop_columns, list)
                and any(isinstance(col, str) for col in drop_columns)
            ):
                context.logger.error(
                    "drop_columns must be an integer/list of integers if not provided with a URI/FeatureVector dataset"
                )
                raise ValueError
            dataset.drop(drop_columns, axis=1, inplace=True)
    else:
        # Dataset is a DataItem with artifact_url (URI or FeatureVector)
        store_uri_prefix, _ = mlrun.datastore.parse_store_uri(dataset.artifact_url)

        # Getting the dataset:
        if mlrun.utils.StorePrefix.FeatureVector == store_uri_prefix:
            label_columns = label_columns or dataset.meta.status.label_column
            context.logger.info(f"label columns: {label_columns}")
            # FeatureVector case:
            try:
                fv = mlrun.datastore.get_store_resource(dataset.artifact_url)
                dataset = fv.get_offline_features(drop_columns=drop_columns).to_dataframe()
            except AttributeError:
                # Leave here for backwards compatibility
                dataset = fs.get_offline_features(
                    dataset.meta.uri, drop_columns=drop_columns
                ).to_dataframe()
        else:
            # simple URL case:
            if not label_columns:
                context.logger.info(
                    "label_columns not provided, mandatory when dataset is not a FeatureVector"
                )
                raise ValueError
            dataset = dataset.as_df()
            if drop_columns:
                if all(col in dataset for col in drop_columns):
                    dataset = dataset.drop(drop_columns, axis=1)
                else:
                    context.logger.info(
                        "not all of the columns to drop in the dataset, drop columns process skipped"
                    )

    return dataset, label_columns


def train(
    context: MLClientCtx,
    dataset: DataItem,
    model_class: str,
    label_columns: Optional[Union[str, List[str]]] = None,
    drop_columns: List[str] = None,
    model_name: str = "model",
    tag: str = "",
    sample_set: DataItem = None,
    test_set: DataItem = None,
    train_test_split_size: float = None,
    random_state: int = None,
    labels: dict = None,
    **kwargs,
):
    """
    Training a model with the given dataset.

    example::

        import mlrun
        project = mlrun.get_or_create_project("my-project")
        project.set_function("hub://auto_trainer", "train")
        trainer_run = project.run(
            name="train",
            handler="train",
            inputs={"dataset": "./path/to/dataset.csv"},
            params={
                "model_class": "sklearn.linear_model.LogisticRegression",
                "label_columns": "label",
                "drop_columns": "id",
                "model_name": "my-model",
                "tag": "v1.0.0",
                "sample_set": "./path/to/sample_set.csv",
                "test_set": "./path/to/test_set.csv",
                "CLASS_solver": "liblinear",
            },
        )

    :param context:                 MLRun context
    :param dataset:                 The dataset to train the model on. Can be either a URI or a FeatureVector
    :param model_class:             The class of the model, e.g. `sklearn.linear_model.LogisticRegression`
    :param label_columns:           The target label(s) of the column(s) in the dataset. for Regression or
                                    Classification tasks. Mandatory when dataset is not a FeatureVector.
    :param drop_columns:            str or a list of strings that represent the columns to drop
    :param model_name:              The model's name to use for storing the model artifact, default to 'model'
    :param tag:                     The model's tag to log with
    :param sample_set:              A sample set of inputs for the model for logging its stats along the model in favour
                                    of model monitoring. Can be either a URI or a FeatureVector
    :param test_set:                The test set to train the model with.
    :param train_test_split_size:   if test_set was provided then this argument is ignored.
                                    Should be between 0.0 and 1.0 and represent the proportion of the dataset to include
                                    in the test split. The size of the Training set is set to the complement of this
                                    value. Default = 0.2
    :param random_state:            Relevant only when using train_test_split_size.
                                    A random state seed to shuffle the data. For more information, see:
                                    https://scikit-learn.org/stable/glossary.html#term-random_state
                                    Notice that here we only pass integer values.
    :param labels:                  Labels to log with the model
    :param kwargs:                  Here you can pass keyword arguments with prefixes,
                                    that will be parsed and passed to the relevant function, by the following prefixes:
                                    - `CLASS_` - for the model class arguments
                                    - `FIT_` - for the `fit` function arguments
                                    - `TRAIN_` - for the `train` function (in xgb or lgbm train function - future)

    """
    # Validate inputs:
    # Check if exactly one of them is supplied:
    if test_set is None:
        if train_test_split_size is None:
            context.logger.info(
                "test_set or train_test_split_size are not provided, setting train_test_split_size to 0.2"
            )
            train_test_split_size = 0.2

    elif train_test_split_size:
        context.logger.info(
            "test_set provided, ignoring given train_test_split_size value"
        )
        train_test_split_size = None

    # Get DataFrame by URL or by FeatureVector:
    dataset, label_columns = _get_dataframe(
        context=context,
        dataset=dataset,
        label_columns=label_columns,
        drop_columns=drop_columns,
    )

    # Getting the sample set:
    if sample_set is None:
        context.logger.info(
            f"Sample set not given, using the whole training set as the sample set"
        )
        sample_set = dataset
    else:
        sample_set, _ = _get_dataframe(
            context=context,
            dataset=sample_set,
            label_columns=label_columns,
            drop_columns=drop_columns,
        )

    # Parsing kwargs:
    # TODO: Use in xgb or lgbm train function.
    train_kwargs = _get_sub_dict_by_prefix(src=kwargs, prefix_key=KWArgsPrefixes.TRAIN)
    fit_kwargs = _get_sub_dict_by_prefix(src=kwargs, prefix_key=KWArgsPrefixes.FIT)
    model_class_kwargs = _get_sub_dict_by_prefix(
        src=kwargs, prefix_key=KWArgsPrefixes.MODEL_CLASS
    )

    # Check if model or function:
    if hasattr(model_class, "train"):
        # TODO: Need to call: model(), afterwards to start the train function.
        # model = create_function(f"{model_class}.train")
        raise NotImplementedError
    else:
        # Creating model instance:
        model = create_class(model_class)(**model_class_kwargs)

    x = dataset.drop(label_columns, axis=1)
    y = dataset[label_columns]
    if train_test_split_size:
        x_train, x_test, y_train, y_test = train_test_split(
            x, y, test_size=train_test_split_size, random_state=random_state
        )
    else:
        x_train, y_train = x, y

        test_set = test_set.as_df()
        if drop_columns:
            test_set = dataset.drop(drop_columns, axis=1)

        x_test, y_test = test_set.drop(label_columns, axis=1), test_set[label_columns]

    AutoMLRun.apply_mlrun(
        model=model,
        model_name=model_name,
        context=context,
        tag=tag,
        sample_set=sample_set,
        y_columns=label_columns,
        test_set=test_set,
        x_test=x_test,
        y_test=y_test,
        artifacts=context.artifacts,
        labels=labels,
    )
    context.logger.info(f"training '{model_name}'")
    model.fit(x_train, y_train, **fit_kwargs)


def evaluate(
    context: MLClientCtx,
    model: str,
    dataset: mlrun.DataItem,
    drop_columns: List[str] = None,
    label_columns: Optional[Union[str, List[str]]] = None,
    **kwargs,
):
    """
    Evaluating a model. Artifacts generated by the MLHandler.

    :param context:                 MLRun context.
    :param model:                   The model Store path.
    :param dataset:                 The dataset to evaluate the model on. Can be either a URI or a FeatureVector.
    :param drop_columns:            str or a list of strings that represent the columns to drop.
    :param label_columns:           The target label(s) of the column(s) in the dataset. for Regression or
                                    Classification tasks. Mandatory when dataset is not a FeatureVector.
    :param kwargs:                  Here you can pass keyword arguments to the predict function
                                    (PREDICT_ prefix is not required).
    """
    # Get dataset by URL or by FeatureVector:
    dataset, label_columns = _get_dataframe(
        context=context,
        dataset=dataset,
        label_columns=label_columns,
        drop_columns=drop_columns,
    )

    # Parsing label_columns:
    parsed_label_columns = []
    if label_columns:
        label_columns = (
            label_columns if isinstance(label_columns, list) else [label_columns]
        )
        for lc in label_columns:
            if fs.common.feature_separator in lc:
                feature_set_name, label_name, alias = fs.common.parse_feature_string(lc)
                parsed_label_columns.append(alias or label_name)
        if parsed_label_columns:
            label_columns = parsed_label_columns

    x = dataset.drop(label_columns, axis=1)
    y = dataset[label_columns]

    # Loading the model and predicting:
    model_handler = AutoMLRun.load_model(
        model_path=model, context=context, model_name="model_LinearRegression"
    )
    AutoMLRun.apply_mlrun(model_handler.model, y_test=y, model_path=model)

    context.logger.info(f"evaluating '{model_handler.model_name}'")
    model_handler.model.predict(x, **kwargs)


def predict(
    context: MLClientCtx,
    model: str,
    dataset: mlrun.DataItem,
    drop_columns: Union[str, List[str], int, List[int]] = None,
    label_columns: Optional[Union[str, List[str]]] = None,
    result_set: Optional[str] = None,
    **kwargs,
):
    """
    Predicting dataset by a model.

    :param context:                 MLRun context.
    :param model:                   The model Store path.
    :param dataset:                 The dataset to predict the model on. Can be either a URI, a FeatureVector or a
                                    sample in a shape of a list/dict.
                                    When passing a sample, pass the dataset as a field in `params` instead of `inputs`.
    :param drop_columns:            str/int or a list of strings/ints that represent the column names/indices to drop.
                                    When the dataset is a list/dict this parameter should be represented by integers.
    :param label_columns:           The target label(s) of the column(s) in the dataset. for Regression or
                                    Classification tasks. Mandatory when dataset is not a FeatureVector.
    :param result_set:              The db key to set name of the prediction result and the filename.
                                    Default to 'prediction'.
    :param kwargs:                  Here you can pass keyword arguments to the predict function
                                    (PREDICT_ prefix is not required).
    """
    # Get dataset by URL or by FeatureVector:
    dataset, label_columns = _get_dataframe(
        context=context,
        dataset=dataset,
        label_columns=label_columns,
        drop_columns=drop_columns,
    )

    # loading the model, and getting the model handler:
    model_handler = AutoMLRun.load_model(model_path=model, context=context)

    # Fix feature names for models that require them (e.g., XGBoost)
    # When dataset comes from a list, pandas assigns default integer column names
    # but some models expect specific feature names they were trained with
    if hasattr(model_handler.model, 'feature_names_in_'):
        expected_features = model_handler.model.feature_names_in_
        if len(dataset.columns) == len(expected_features):
            # Only rename if the number of columns matches
            # This handles the case where a list was converted to DataFrame with default column names
            if not all(col == feat for col, feat in zip(dataset.columns, expected_features)):
                context.logger.info(
                    f"Renaming dataset columns to match model's expected feature names"
                )
                dataset.columns = expected_features

    # Dropping label columns if necessary:
    if not label_columns:
        label_columns = []
    elif isinstance(label_columns, str):
        label_columns = [label_columns]

    # Predicting:
    context.logger.info(f"making prediction by '{model_handler.model_name}'")
    y_pred = model_handler.model.predict(dataset, **kwargs)

    # Preparing and validating label columns for the dataframe of the prediction result:
    num_predicted = 1 if len(y_pred.shape) == 1 else y_pred.shape[1]

    if num_predicted > len(label_columns):
        if num_predicted == 1:
            label_columns = ["predicted labels"]
        else:
            label_columns.extend(
                [
                    f"predicted_label_{i + 1 + len(label_columns)}"
                    for i in range(num_predicted - len(label_columns))
                ]
            )
    elif num_predicted < len(label_columns):
        context.logger.error(
            f"number of predicted labels: {num_predicted} is smaller than number of label columns: {len(label_columns)}"
        )
        raise ValueError

    artifact_name = result_set or "prediction"
    labels_inside_df = set(label_columns) & set(dataset.columns.tolist())
    if labels_inside_df:
        context.logger.error(
            f"The labels: {labels_inside_df} are already existed in the dataframe"
        )
        raise ValueError
    pred_df = pd.concat([dataset, pd.DataFrame(y_pred, columns=label_columns)], axis=1)
    context.log_dataset(artifact_name, pred_df, db_key=result_set)
 command: '' -kind: job -verbose: false + default_handler: train + image: mlrun/mlrun + disable_auto_mount: false + description: Automatic train, evaluate and predict functions for the ML frameworks + - Scikit-Learn, XGBoost and LightGBM. +metadata: + categories: + - machine-learning + - model-training + tag: '' + name: auto-trainer diff --git a/functions/src/auto_trainer/item.yaml b/functions/src/auto_trainer/item.yaml index ba33f6a08..d397a79d6 100755 --- a/functions/src/auto_trainer/item.yaml +++ b/functions/src/auto_trainer/item.yaml @@ -13,7 +13,7 @@ labels: author: Iguazio maintainers: [] marketplaceType: '' -mlrunVersion: 1.7.0 +mlrunVersion: 1.10.0 name: auto_trainer platformVersion: 3.5.0 spec: @@ -23,4 +23,4 @@ spec: kind: job requirements: [] url: '' -version: 1.8.0 +version: 1.9.0 diff --git a/functions/src/auto_trainer/requirements.txt b/functions/src/auto_trainer/requirements.txt index b14a0293c..b23f9b9dd 100644 --- a/functions/src/auto_trainer/requirements.txt +++ b/functions/src/auto_trainer/requirements.txt @@ -1,4 +1,5 @@ pandas -scikit-learn<1.4.0 +scikit-learn~=1.5.2 +lightgbm xgboost<2.0.0 plotly diff --git a/functions/src/auto_trainer/test_auto_trainer.py b/functions/src/auto_trainer/test_auto_trainer.py index 9a1ff554c..ac95109f8 100644 --- a/functions/src/auto_trainer/test_auto_trainer.py +++ b/functions/src/auto_trainer/test_auto_trainer.py @@ -29,6 +29,9 @@ ("sklearn.linear_model.LinearRegression", "regression"), ("sklearn.ensemble.RandomForestClassifier", "classification"), ("xgboost.XGBRegressor", "regression"), + ("xgboost.XGBClassifier", "classification"), + ("lightgbm.LGBMRegressor", "regression"), + ("lightgbm.LGBMClassifier", "classification") ] REQUIRED_ENV_VARS = [ @@ -78,11 +81,15 @@ def _assert_train_handler(train_run): @pytest.mark.parametrize("model", MODELS) +@pytest.mark.skipif( + condition=not _validate_environment_variables(), + reason="Project's environment variables are not set", +) def test_train(model: Tuple[str, str]): dataset, label_columns = _get_dataset(model[1]) is_test_passed = True - project = mlrun.new_project("auto-trainer-test", context="./") + project = mlrun.get_or_create_project("auto-trainer-test", context="./") fn = project.set_function("function.yaml", "train", kind="job", image="mlrun/mlrun") train_run = None @@ -119,7 +126,7 @@ def test_train_evaluate(model: Tuple[str, str]): dataset, label_columns = _get_dataset(model[1]) is_test_passed = True # Importing function: - project = mlrun.new_project("auto-trainer-test", context="./") + project = mlrun.get_or_create_project("auto-trainer-test", context="./") fn = project.set_function("function.yaml", "train", kind="job", image="mlrun/mlrun") temp_dir = tempfile.mkdtemp() @@ -172,7 +179,7 @@ def test_train_predict(model: Tuple[str, str]): df = pd.read_csv(dataset) sample = df.head().drop("labels", axis=1).values.tolist() # Importing function: - project = mlrun.new_project("auto-trainer-test", context="./") + project = mlrun.get_or_create_project("auto-trainer-test", context="./") fn = project.set_function("function.yaml", "train", kind="job", image="mlrun/mlrun") temp_dir = tempfile.mkdtemp() diff --git a/functions/src/describe/function.yaml b/functions/src/describe/function.yaml index a11461774..1c254c3c4 100644 --- a/functions/src/describe/function.yaml +++ b/functions/src/describe/function.yaml @@ -1,9 +1,44 @@ +metadata: + tag: '' + categories: + - data-analysis + name: describe +verbose: false +kind: job spec: + command: '' + image: mlrun/mlrun + description: describe and visualizes dataset stats + disable_auto_mount: false + default_handler: analyze entry_points: analyze: + doc: 'The function will output the following artifacts per + + column within the data frame (based on data types) + + If the data has more than 500,000 sample we + + sample randomly 500,000 samples: + + + describe csv + + histograms + + scatter-2d + + violin chart + + correlation-matrix chart + + correlation-matrix csv + + imbalance pie chart + + imbalance-weights-vec csv' + has_kwargs: false has_varargs: false - outputs: - - type: None parameters: - name: context type: MLClientCtx @@ -45,46 +80,11 @@ spec: - name: dask_client doc: Dask client object default: null - doc: 'The function will output the following artifacts per - - column within the data frame (based on data types) - - If the data has more than 500,000 sample we - - sample randomly 500,000 samples: - - - describe csv - - histograms - - scatter-2d - - violin chart - - correlation-matrix chart - - correlation-matrix csv - - imbalance pie chart - - imbalance-weights-vec csv' - has_kwargs: false + outputs: + - type: None name: analyze lineno: 46 - image: mlrun/mlrun - command: '' build: - functionSourceCode: # Copyright 2019 Iguazio
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# Generated by nuclio.export.NuclioExporter

import warnings
from typing import Union

import mlrun
import numpy as np

warnings.simplefilter(action="ignore", category=FutureWarning)

import mlrun.feature_store as fstore
import pandas as pd
import plotly.express as px
import plotly.figure_factory as ff
import plotly.graph_objects as go
from mlrun.artifacts import (
    Artifact,
    DatasetArtifact,
    PlotlyArtifact,
    TableArtifact,
    update_dataset_meta,
)
from mlrun.datastore import DataItem
from mlrun.execution import MLClientCtx
from mlrun.feature_store import FeatureSet
from plotly.subplots import make_subplots

pd.set_option("display.float_format", lambda x: "%.2f" % x)
MAX_SIZE_OF_DF = 500000


def analyze(
    context: MLClientCtx,
    name: str = "dataset",
    table: Union[FeatureSet, DataItem] = None,
    label_column: str = None,
    plots_dest: str = "plots",
    random_state: int = 1,
    problem_type: str = "classification",
    dask_key: str = "dask_key",
    dask_function: str = None,
    dask_client=None,
) -> None:
    """
    The function will output the following artifacts per
    column within the data frame (based on data types)
    If the data has more than 500,000 sample we
    sample randomly 500,000 samples:

    describe csv
    histograms
    scatter-2d
    violin chart
    correlation-matrix chart
    correlation-matrix csv
    imbalance pie chart
    imbalance-weights-vec csv

    :param context:                 The function context
    :param name:                    Key of dataset to database ("dataset" for default)
    :param table:                   MLRun input pointing to pandas dataframe (csv/parquet file path) or FeatureSet
                                    as param
    :param label_column:            Ground truth column label
    :param plots_dest:              Destination folder of summary plots (relative to artifact_path)
                                    ("plots" for default)
    :param random_state:            When the table has more than 500,000 samples, we sample randomly 500,000 samples
    :param problem_type             The type of the ML problem the data facing - regression, classification or None
                                    (classification for default)
    :param dask_key:                Key of dataframe in dask client "datasets" attribute
    :param dask_function:           Dask function url (db://..)
    :param dask_client:             Dask client object
    """
    data_item, featureset, creat, update = False, False, False, False
    get_from_table = True
    if dask_function or dask_client:
        data_item, creat = True, True
        if dask_function:
            client = mlrun.import_function(dask_function).client
        elif dask_client:
            client = dask_client
        else:
            raise ValueError("dask client was not provided")

        if dask_key in client.datasets:
            df = client.get_dataset(dask_key)
            data_item, creat, get_from_table = True, True, False
        elif table:
            get_from_table = True
        else:
            context.logger.info(
                f"only these datasets are available {client.datasets} in client {client}"
            )
            raise Exception("dataset not found on dask cluster")

    if get_from_table:
        if type(table) == DataItem:
            if table.meta is None:
                data_item, creat, update = True, True, False
            elif table.meta.kind == "dataset":
                data_item, creat, update = True, False, True
            elif table.meta.kind == "FeatureVector":
                data_item, creat, update = True, False, False
            elif table.meta.kind == "FeatureSet":
                featureset, creat, update = True, False, False

        if data_item:
            df = table.as_df()
        elif featureset:
            project_name, set_name = (
                table._path.split("/")[2],
                table._path.split("/")[4],
            )
            feature_set = fstore.get_feature_set(
                f"store://feature-sets/{project_name}/{set_name}"
            )
            df = feature_set.to_dataframe()
        else:
            context.logger.error(f"Wrong table type.")
            return

    if df.size > MAX_SIZE_OF_DF:
        df = df.sample(n=int(MAX_SIZE_OF_DF / df.shape[1]), random_state=random_state)
    extra_data = {}

    if label_column not in df.columns:
        label_column = None

    extra_data["describe csv"] = context.log_artifact(
        TableArtifact("describe-csv", df=df.describe()),
        local_path=f"{plots_dest}/describe.csv",
    )

    try:
        _create_histogram_mat_artifact(
            context, df, extra_data, label_column, plots_dest
        )
    except Exception as e:
        context.logger.warn(f"Failed to create histogram matrix artifact due to: {e}")
    try:
        _create_features_histogram_artifacts(
            context, df, extra_data, label_column, plots_dest, problem_type
        )
    except Exception as e:
        context.logger.warn(f"Failed to create pairplot histograms due to: {e}")
    try:
        _create_features_2d_scatter_artifacts(
            context, df, extra_data, label_column, plots_dest, problem_type
        )
    except Exception as e:
        context.logger.warn(f"Failed to create pairplot 2d_scatter due to: {e}")
    try:
        _create_violin_artifact(context, df, extra_data, plots_dest)
    except Exception as e:
        context.logger.warn(f"Failed to create violin distribution plots due to: {e}")
    try:
        _create_imbalance_artifact(
            context, df, extra_data, label_column, plots_dest, problem_type
        )
    except Exception as e:
        context.logger.warn(f"Failed to create class imbalance plot due to: {e}")
    try:
        _create_corr_artifact(context, df, extra_data, label_column, plots_dest)
    except Exception as e:
        context.logger.warn(f"Failed to create features correlation plot due to: {e}")

    if not data_item:
        return

    artifact = table.artifact_url
    if creat:  # dataset not stored
        artifact = DatasetArtifact(
            key="dataset", stats=True, df=df, extra_data=extra_data
        )
        artifact = context.log_artifact(artifact, db_key=name)
        context.logger.info(f"The data set is logged to the project under {name} name")

    if update:
        update_dataset_meta(artifact, extra_data=extra_data)
        context.logger.info(f"The data set named {name} is updated")

    # TODO : 3-D plot on on selected features.
    # TODO : Reintegration plot on on selected features.
    # TODO : PCA plot (with options)


def _create_histogram_mat_artifact(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
):
    """
    Create and log a histogram matrix artifact
    """
    context.log_artifact(
        item=Artifact(
            key="hist",
            body=b"<b> Deprecated, see the artifacts scatter-2d "
            b"and histograms instead<b>",
        ),
        local_path=f"{plots_dest}/hist.html",
    )


def _create_features_histogram_artifacts(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
    problem_type: str,
):
    """
    Create and log a histogram artifact for each feature
    """

    figs = dict()
    first_feature_name = ""
    if label_column is not None and problem_type == "classification":
        all_labels = df[label_column].unique()
    visible = True
    for column_name in df.columns:
        if column_name == label_column:
            continue

        if label_column is not None and problem_type == "classification":
            for label in all_labels:
                sub_fig = go.Histogram(
                    histfunc="count",
                    x=df.loc[df[label_column] == label][column_name],
                    name=str(label),
                    visible=visible,
                )
                figs[f"{column_name}@?@{label}"] = sub_fig
        else:
            sub_fig = go.Histogram(histfunc="count", x=df[column_name], visible=visible)
            figs[f"{column_name}@?@{1}"] = sub_fig
        if visible:
            first_feature_name = column_name
        visible = False

    fig = go.Figure()
    for k in figs.keys():
        fig.add_trace(figs[k])

    fig.update_layout(
        updatemenus=[
            {
                "buttons": [
                    {
                        "label": column_name,
                        "method": "update",
                        "args": [
                            {
                                "visible": [
                                    key.split("@?@")[0] == column_name
                                    for key in figs.keys()
                                ],
                                "xaxis": {
                                    "range": [
                                        min(df[column_name]),
                                        max(df[column_name]),
                                    ]
                                },
                            },
                            {"title": f"<i><b>Histogram of {column_name}</b></i>"},
                        ],
                    }
                    for column_name in df.columns
                    if column_name != label_column
                ],
                "direction": "down",
                "pad": {"r": 10, "t": 10},
                "showactive": True,
                "x": 0.25,
                "xanchor": "left",
                "y": 1.1,
                "yanchor": "top",
            }
        ],
        annotations=[
            dict(
                text="Select Feature Name ",
                showarrow=False,
                x=0,
                y=1.05,
                yref="paper",
                xref="paper",
                align="left",
                xanchor="left",
                yanchor="top",
                font={
                    "color": "blue",
                },
            )
        ],
    )

    fig.update_layout(
        width=600,
        height=400,
        autosize=False,
        margin=dict(t=100, b=0, l=0, r=0),
        template="plotly_white",
    )

    fig.update_layout(title_text=f"<i><b>Histograms of {first_feature_name}</b></i>")
    extra_data[f"histograms"] = context.log_artifact(
        PlotlyArtifact(key=f"histograms", figure=fig),
        local_path=f"{plots_dest}/histograms.html",
    )


def _create_features_2d_scatter_artifacts(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
    problem_type: str,
):
    """
    Create and log a scatter-2d artifact for each couple of features
    """
    features = [
        column_name for column_name in df.columns if column_name != label_column
    ]
    max_feature_len = float(max(len(elem) for elem in features))
    if label_column is not None:
        labels = sorted(df[label_column].unique())
    else:
        labels = [None]
    fig = go.Figure()
    if label_column is not None and problem_type == "classification":
        for l in labels:
            fig.add_trace(
                go.Scatter(
                    x=df.loc[df[label_column] == l][features[0]],
                    y=df.loc[df[label_column] == l][features[0]],
                    mode="markers",
                    visible=True,
                    showlegend=True,
                    name=str(l),
                )
            )
    elif label_column is None:
        fig.add_trace(
            go.Scatter(
                x=df[features[0]],
                y=df[features[0]],
                mode="markers",
                visible=True,
            )
        )
    elif problem_type == "regression":
        fig.add_trace(
            go.Scatter(
                x=df[features[0]],
                y=df[features[0]],
                mode="markers",
                marker=dict(
                    color=df[label_column], colorscale="Viridis", showscale=True
                ),
                visible=True,
            )
        )

    x_buttons = []
    y_buttons = []

    for ncol in features:
        if problem_type == "classification" and label_column is not None:
            x_buttons.append(
                dict(
                    method="update",
                    label=ncol,
                    args=[
                        {"x": [df.loc[df[label_column] == l][ncol] for l in labels]},
                        np.arange(len(labels)).tolist(),
                    ],
                )
            )

            y_buttons.append(
                dict(
                    method="update",
                    label=ncol,
                    args=[
                        {"y": [df.loc[df[label_column] == l][ncol] for l in labels]},
                        np.arange(len(labels)).tolist(),
                    ],
                )
            )
        else:
            x_buttons.append(
                dict(method="update", label=ncol, args=[{"x": [df[ncol]]}])
            )

            y_buttons.append(
                dict(method="update", label=ncol, args=[{"y": [df[ncol]]}])
            )

    # Pass buttons to the updatemenus argument
    fig.update_layout(
        updatemenus=[
            dict(buttons=x_buttons, direction="up", x=0.5, y=-0.1),
            dict(buttons=y_buttons, direction="down", x=-max_feature_len / 100, y=0.5),
        ]
    )

    fig.update_layout(
        width=600,
        height=400,
        autosize=False,
        margin=dict(t=100, b=0, l=0, r=0),
        template="plotly_white",
    )

    fig.update_layout(title_text=f"<i><b>Scatter-2d</b></i>")
    extra_data[f"scatter-2d"] = context.log_artifact(
        PlotlyArtifact(key=f"scatter-2d", figure=fig),
        local_path=f"{plots_dest}/scatter-2d.html",
    )


def _create_violin_artifact(
    context: MLClientCtx, df: pd.DataFrame, extra_data: dict, plots_dest: str
):
    """
    Create and log a violin artifact
    """
    cols = 5
    rows = (df.shape[1] // cols) + 1
    fig = make_subplots(rows=rows, cols=cols)

    plot_num = 0

    for column_name in df.columns:
        column_data = df[column_name]
        violin = go.Violin(
            x=[column_name] * column_data.shape[0],
            y=column_data,
            name=column_name,
        )

        fig.add_trace(
            violin,
            row=(plot_num // cols) + 1,
            col=(plot_num % cols) + 1,
        )

        plot_num += 1

    fig["layout"].update(
        height=(rows + 1) * 200,
        width=(cols + 1) * 200,
        title="<i><b>Violin Plots</b></i>",
    )

    fig.update_layout(showlegend=False)
    extra_data["violin"] = context.log_artifact(
        PlotlyArtifact(key="violin", figure=fig),
        local_path=f"{plots_dest}/violin.html",
    )


def _create_imbalance_artifact(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
    problem_type: str,
):
    """
    Create and log an imbalance class artifact (csv + plot)
    """
    if label_column:
        if problem_type == "classification":
            values_column = "count"
            labels_count = df[label_column].value_counts().sort_index()
            df_labels_count = pd.DataFrame(labels_count)
            df_labels_count[label_column] = labels_count.index
            df_labels_count.rename(columns={"": values_column}, inplace=True)
            df_labels_count[values_column] = df_labels_count[values_column] / sum(
                df_labels_count[values_column]
            )
            fig = px.pie(df_labels_count, names=label_column, values=values_column)
        else:
            fig = px.histogram(
                histfunc="count",
                x=df[label_column],
            )
            hist = np.histogram(df[label_column])
            df_labels_count = pd.DataFrame(
                {"min_val": hist[1], "count": hist[0].tolist() + [0]}
            )
        fig.update_layout(title_text="<i><b>Labels Imbalance</b></i>")
        extra_data["imbalance"] = context.log_artifact(
            PlotlyArtifact(key="imbalance", figure=fig),
            local_path=f"{plots_dest}/imbalance.html",
        )
        extra_data["imbalance-csv"] = context.log_artifact(
            TableArtifact("imbalance-weights-vec", df=df_labels_count),
            local_path=f"{plots_dest}/imbalance-weights-vec.csv",
        )


def _create_corr_artifact(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
):
    """
    Create and log an correlation-matrix artifact (csv + plot)
    """
    if label_column is not None:
        df = df.drop([label_column], axis=1)
    tblcorr = df.corr(numeric_only=True)
    extra_data["correlation-matrix-csv"] = context.log_artifact(
        TableArtifact("correlation-matrix-csv", df=tblcorr, visible=True),
        local_path=f"{plots_dest}/correlation-matrix.csv",
    )

    z = tblcorr.values.tolist()
    z_text = [["{:.2f}".format(y) for y in x] for x in z]
    fig = ff.create_annotated_heatmap(
        z,
        x=list(tblcorr.columns),
        y=list(tblcorr.columns),
        annotation_text=z_text,
        colorscale="agsunset",
    )
    fig["layout"]["yaxis"]["autorange"] = "reversed"  # l -> r
    fig.update_layout(title_text="<i><b>Correlation matrix</b></i>")
    fig["data"][0]["showscale"] = True

    extra_data["correlation"] = context.log_artifact(
        PlotlyArtifact(key="correlation", figure=fig),
        local_path=f"{plots_dest}/correlation.html",
    )
 - code_origin: '' origin_filename: '' - description: describe and visualizes dataset stats - disable_auto_mount: false - default_handler: analyze -verbose: false -metadata: - tag: '' - name: describe - categories: - - data-analysis -kind: job + code_origin: '' + functionSourceCode: # Copyright 2019 Iguazio
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# Generated by nuclio.export.NuclioExporter

import warnings
from typing import Union

import mlrun
import numpy as np

warnings.simplefilter(action="ignore", category=FutureWarning)

import mlrun.feature_store as fstore
import pandas as pd
import plotly.express as px
import plotly.figure_factory as ff
import plotly.graph_objects as go
from mlrun.artifacts import (
    Artifact,
    DatasetArtifact,
    PlotlyArtifact,
    TableArtifact,
    update_dataset_meta,
)
from mlrun.datastore import DataItem
from mlrun.execution import MLClientCtx
from mlrun.feature_store import FeatureSet
from plotly.subplots import make_subplots

pd.set_option("display.float_format", lambda x: "%.2f" % x)
MAX_SIZE_OF_DF = 500000


def analyze(
    context: MLClientCtx,
    name: str = "dataset",
    table: Union[FeatureSet, DataItem] = None,
    label_column: str = None,
    plots_dest: str = "plots",
    random_state: int = 1,
    problem_type: str = "classification",
    dask_key: str = "dask_key",
    dask_function: str = None,
    dask_client=None,
) -> None:
    """
    The function will output the following artifacts per
    column within the data frame (based on data types)
    If the data has more than 500,000 sample we
    sample randomly 500,000 samples:

    describe csv
    histograms
    scatter-2d
    violin chart
    correlation-matrix chart
    correlation-matrix csv
    imbalance pie chart
    imbalance-weights-vec csv

    :param context:                 The function context
    :param name:                    Key of dataset to database ("dataset" for default)
    :param table:                   MLRun input pointing to pandas dataframe (csv/parquet file path) or FeatureSet
                                    as param
    :param label_column:            Ground truth column label
    :param plots_dest:              Destination folder of summary plots (relative to artifact_path)
                                    ("plots" for default)
    :param random_state:            When the table has more than 500,000 samples, we sample randomly 500,000 samples
    :param problem_type             The type of the ML problem the data facing - regression, classification or None
                                    (classification for default)
    :param dask_key:                Key of dataframe in dask client "datasets" attribute
    :param dask_function:           Dask function url (db://..)
    :param dask_client:             Dask client object
    """
    data_item, featureset, creat, update = False, False, False, False
    get_from_table = True
    if dask_function or dask_client:
        data_item, creat = True, True
        if dask_function:
            client = mlrun.import_function(dask_function).client
        elif dask_client:
            client = dask_client
        else:
            raise ValueError("dask client was not provided")

        if dask_key in client.datasets:
            df = client.get_dataset(dask_key)
            data_item, creat, get_from_table = True, True, False
        elif table:
            get_from_table = True
        else:
            context.logger.info(
                f"only these datasets are available {client.datasets} in client {client}"
            )
            raise Exception("dataset not found on dask cluster")

    if get_from_table:
        if type(table) == DataItem:
            if table.meta is None:
                data_item, creat, update = True, True, False
            elif table.meta.kind == "dataset":
                data_item, creat, update = True, False, True
            elif table.meta.kind == "FeatureVector":
                data_item, creat, update = True, False, False
            elif table.meta.kind == "FeatureSet":
                featureset, creat, update = True, False, False

        if data_item:
            df = table.as_df()
        elif featureset:
            project_name, set_name = (
                table._path.split("/")[2],
                table._path.split("/")[4],
            )
            feature_set = fstore.get_feature_set(
                f"store://feature-sets/{project_name}/{set_name}"
            )
            df = feature_set.to_dataframe()
        else:
            context.logger.error(f"Wrong table type.")
            return

    if df.size > MAX_SIZE_OF_DF:
        df = df.sample(n=int(MAX_SIZE_OF_DF / df.shape[1]), random_state=random_state)
    extra_data = {}

    if label_column not in df.columns:
        label_column = None

    extra_data["describe csv"] = context.log_artifact(
        TableArtifact("describe-csv", df=df.describe()),
        local_path=f"{plots_dest}/describe.csv",
    )

    try:
        _create_histogram_mat_artifact(
            context, df, extra_data, label_column, plots_dest
        )
    except Exception as e:
        context.logger.warn(f"Failed to create histogram matrix artifact due to: {e}")
    try:
        _create_features_histogram_artifacts(
            context, df, extra_data, label_column, plots_dest, problem_type
        )
    except Exception as e:
        context.logger.warn(f"Failed to create pairplot histograms due to: {e}")
    try:
        _create_features_2d_scatter_artifacts(
            context, df, extra_data, label_column, plots_dest, problem_type
        )
    except Exception as e:
        context.logger.warn(f"Failed to create pairplot 2d_scatter due to: {e}")
    try:
        _create_violin_artifact(context, df, extra_data, plots_dest)
    except Exception as e:
        context.logger.warn(f"Failed to create violin distribution plots due to: {e}")
    try:
        _create_imbalance_artifact(
            context, df, extra_data, label_column, plots_dest, problem_type
        )
    except Exception as e:
        context.logger.warn(f"Failed to create class imbalance plot due to: {e}")
    try:
        _create_corr_artifact(context, df, extra_data, label_column, plots_dest)
    except Exception as e:
        context.logger.warn(f"Failed to create features correlation plot due to: {e}")

    if not data_item:
        return

    artifact = table.artifact_url
    if creat:  # dataset not stored
        artifact = DatasetArtifact(
            key="dataset", stats=True, df=df, extra_data=extra_data
        )
        artifact = context.log_artifact(artifact, db_key=name)
        context.logger.info(f"The data set is logged to the project under {name} name")

    if update:
        update_dataset_meta(artifact, extra_data=extra_data)
        context.logger.info(f"The data set named {name} is updated")

    # TODO : 3-D plot on on selected features.
    # TODO : Reintegration plot on on selected features.
    # TODO : PCA plot (with options)


def _create_histogram_mat_artifact(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
):
    """
    Create and log a histogram matrix artifact
    """
    context.log_artifact(
        item=Artifact(
            key="hist",
            body=b"<b> Deprecated, see the artifacts scatter-2d "
            b"and histograms instead<b>",
        ),
        local_path=f"{plots_dest}/hist.html",
    )


def _create_features_histogram_artifacts(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
    problem_type: str,
):
    """
    Create and log a histogram artifact for each feature
    """

    figs = dict()
    first_feature_name = ""
    if label_column is not None and problem_type == "classification":
        all_labels = df[label_column].unique()
    visible = True
    for column_name in df.columns:
        if column_name == label_column:
            continue

        if label_column is not None and problem_type == "classification":
            for label in all_labels:
                sub_fig = go.Histogram(
                    histfunc="count",
                    x=df.loc[df[label_column] == label][column_name],
                    name=str(label),
                    visible=visible,
                )
                figs[f"{column_name}@?@{label}"] = sub_fig
        else:
            sub_fig = go.Histogram(histfunc="count", x=df[column_name], visible=visible)
            figs[f"{column_name}@?@{1}"] = sub_fig
        if visible:
            first_feature_name = column_name
        visible = False

    fig = go.Figure()
    for k in figs.keys():
        fig.add_trace(figs[k])

    fig.update_layout(
        updatemenus=[
            {
                "buttons": [
                    {
                        "label": column_name,
                        "method": "update",
                        "args": [
                            {
                                "visible": [
                                    key.split("@?@")[0] == column_name
                                    for key in figs.keys()
                                ],
                                "xaxis": {
                                    "range": [
                                        min(df[column_name]),
                                        max(df[column_name]),
                                    ]
                                },
                            },
                            {"title": f"<i><b>Histogram of {column_name}</b></i>"},
                        ],
                    }
                    for column_name in df.columns
                    if column_name != label_column
                ],
                "direction": "down",
                "pad": {"r": 10, "t": 10},
                "showactive": True,
                "x": 0.25,
                "xanchor": "left",
                "y": 1.1,
                "yanchor": "top",
            }
        ],
        annotations=[
            dict(
                text="Select Feature Name ",
                showarrow=False,
                x=0,
                y=1.05,
                yref="paper",
                xref="paper",
                align="left",
                xanchor="left",
                yanchor="top",
                font={
                    "color": "blue",
                },
            )
        ],
    )

    fig.update_layout(
        width=600,
        height=400,
        autosize=False,
        margin=dict(t=100, b=0, l=0, r=0),
        template="plotly_white",
    )

    fig.update_layout(title_text=f"<i><b>Histograms of {first_feature_name}</b></i>")
    extra_data[f"histograms"] = context.log_artifact(
        PlotlyArtifact(key=f"histograms", figure=fig),
        local_path=f"{plots_dest}/histograms.html",
    )


def _create_features_2d_scatter_artifacts(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
    problem_type: str,
):
    """
    Create and log a scatter-2d artifact for each couple of features
    """
    features = [
        column_name for column_name in df.columns if column_name != label_column
    ]
    max_feature_len = float(max(len(elem) for elem in features))
    if label_column is not None:
        labels = sorted(df[label_column].unique())
    else:
        labels = [None]
    fig = go.Figure()
    if label_column is not None and problem_type == "classification":
        for l in labels:
            fig.add_trace(
                go.Scatter(
                    x=df.loc[df[label_column] == l][features[0]],
                    y=df.loc[df[label_column] == l][features[0]],
                    mode="markers",
                    visible=True,
                    showlegend=True,
                    name=str(l),
                )
            )
    elif label_column is None:
        fig.add_trace(
            go.Scatter(
                x=df[features[0]],
                y=df[features[0]],
                mode="markers",
                visible=True,
            )
        )
    elif problem_type == "regression":
        fig.add_trace(
            go.Scatter(
                x=df[features[0]],
                y=df[features[0]],
                mode="markers",
                marker=dict(
                    color=df[label_column], colorscale="Viridis", showscale=True
                ),
                visible=True,
            )
        )

    x_buttons = []
    y_buttons = []

    for ncol in features:
        if problem_type == "classification" and label_column is not None:
            x_buttons.append(
                dict(
                    method="update",
                    label=ncol,
                    args=[
                        {"x": [df.loc[df[label_column] == l][ncol] for l in labels]},
                        np.arange(len(labels)).tolist(),
                    ],
                )
            )

            y_buttons.append(
                dict(
                    method="update",
                    label=ncol,
                    args=[
                        {"y": [df.loc[df[label_column] == l][ncol] for l in labels]},
                        np.arange(len(labels)).tolist(),
                    ],
                )
            )
        else:
            x_buttons.append(
                dict(method="update", label=ncol, args=[{"x": [df[ncol]]}])
            )

            y_buttons.append(
                dict(method="update", label=ncol, args=[{"y": [df[ncol]]}])
            )

    # Pass buttons to the updatemenus argument
    fig.update_layout(
        updatemenus=[
            dict(buttons=x_buttons, direction="up", x=0.5, y=-0.1),
            dict(buttons=y_buttons, direction="down", x=-max_feature_len / 100, y=0.5),
        ]
    )

    fig.update_layout(
        width=600,
        height=400,
        autosize=False,
        margin=dict(t=100, b=0, l=0, r=0),
        template="plotly_white",
    )

    fig.update_layout(title_text=f"<i><b>Scatter-2d</b></i>")
    extra_data[f"scatter-2d"] = context.log_artifact(
        PlotlyArtifact(key=f"scatter-2d", figure=fig),
        local_path=f"{plots_dest}/scatter-2d.html",
    )


def _create_violin_artifact(
    context: MLClientCtx, df: pd.DataFrame, extra_data: dict, plots_dest: str
):
    """
    Create and log a violin artifact
    """
    cols = 5
    rows = (df.shape[1] // cols) + 1
    fig = make_subplots(rows=rows, cols=cols)

    plot_num = 0

    for column_name in df.columns:
        column_data = df[column_name]
        violin = go.Violin(
            x=[column_name] * column_data.shape[0],
            y=column_data,
            name=column_name,
        )

        fig.add_trace(
            violin,
            row=(plot_num // cols) + 1,
            col=(plot_num % cols) + 1,
        )

        plot_num += 1

    fig["layout"].update(
        height=(rows + 1) * 200,
        width=(cols + 1) * 200,
        title="<i><b>Violin Plots</b></i>",
    )

    fig.update_layout(showlegend=False)
    extra_data["violin"] = context.log_artifact(
        PlotlyArtifact(key="violin", figure=fig),
        local_path=f"{plots_dest}/violin.html",
    )


def _create_imbalance_artifact(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
    problem_type: str,
):
    """
    Create and log an imbalance class artifact (csv + plot)
    """
    if label_column:
        if problem_type == "classification":
            values_column = "count"
            labels_count = df[label_column].value_counts().sort_index()
            df_labels_count = pd.DataFrame(labels_count)
            df_labels_count[label_column] = labels_count.index
            df_labels_count.rename(columns={"": values_column}, inplace=True)
            df_labels_count[values_column] = df_labels_count[values_column] / sum(
                df_labels_count[values_column]
            )
            fig = px.pie(df_labels_count, names=label_column, values=values_column)
        else:
            fig = px.histogram(
                histfunc="count",
                x=df[label_column],
            )
            hist = np.histogram(df[label_column])
            df_labels_count = pd.DataFrame(
                {"min_val": hist[1], "count": hist[0].tolist() + [0]}
            )
        fig.update_layout(title_text="<i><b>Labels Imbalance</b></i>")
        extra_data["imbalance"] = context.log_artifact(
            PlotlyArtifact(key="imbalance", figure=fig),
            local_path=f"{plots_dest}/imbalance.html",
        )
        extra_data["imbalance-csv"] = context.log_artifact(
            TableArtifact("imbalance-weights-vec", df=df_labels_count),
            local_path=f"{plots_dest}/imbalance-weights-vec.csv",
        )


def _create_corr_artifact(
    context: MLClientCtx,
    df: pd.DataFrame,
    extra_data: dict,
    label_column: str,
    plots_dest: str,
):
    """
    Create and log an correlation-matrix artifact (csv + plot)
    """
    if label_column is not None:
        df = df.drop([label_column], axis=1)
    tblcorr = df.corr(numeric_only=True)
    extra_data["correlation-matrix-csv"] = context.log_artifact(
        TableArtifact("correlation-matrix-csv", df=tblcorr, visible=True),
        local_path=f"{plots_dest}/correlation-matrix.csv",
    )

    z = tblcorr.values.tolist()
    z_text = [["{:.2f}".format(y) for y in x] for x in z]
    fig = ff.create_annotated_heatmap(
        z,
        x=list(tblcorr.columns),
        y=list(tblcorr.columns),
        annotation_text=z_text,
        colorscale="agsunset",
    )
    fig["layout"]["yaxis"]["autorange"] = "reversed"  # l -> r
    fig.update_layout(title_text="<i><b>Correlation matrix</b></i>")
    fig["data"][0]["showscale"] = True

    extra_data["correlation"] = context.log_artifact(
        PlotlyArtifact(key="correlation", figure=fig),
        local_path=f"{plots_dest}/correlation.html",
    )
 diff --git a/functions/src/describe/item.yaml b/functions/src/describe/item.yaml index da26f1501..a1aa47372 100644 --- a/functions/src/describe/item.yaml +++ b/functions/src/describe/item.yaml @@ -11,7 +11,7 @@ labels: author: Iguazio maintainers: [] marketplaceType: '' -mlrunVersion: 1.7.0 +mlrunVersion: 1.10.0 name: describe platformVersion: 3.5.3 spec: @@ -21,4 +21,4 @@ spec: kind: job requirements: [] url: '' -version: 1.4.0 +version: 1.5.0 diff --git a/functions/src/describe/requirements.txt b/functions/src/describe/requirements.txt index 15492b176..ac445e6d6 100644 --- a/functions/src/describe/requirements.txt +++ b/functions/src/describe/requirements.txt @@ -1,4 +1,4 @@ -scikit-learn~=1.0.2 +scikit-learn~=1.5.2 plotly~=5.23 pytest~=7.0.1 matplotlib~=3.5.1 diff --git a/functions/src/gen_class_data/function.yaml b/functions/src/gen_class_data/function.yaml index 1769bec07..fa802964e 100644 --- a/functions/src/gen_class_data/function.yaml +++ b/functions/src/gen_class_data/function.yaml @@ -1,13 +1,15 @@ metadata: - categories: - - data-generation tag: '' name: gen-class-data + categories: + - data-generation +verbose: false spec: description: Create a binary classification sample dataset and save. - default_handler: gen_class_data entry_points: gen_class_data: + lineno: 22 + has_varargs: false has_kwargs: false parameters: - name: context @@ -48,7 +50,6 @@ spec: - name: sk_params doc: additional parameters for `sklearn.datasets.make_classification` default: {} - lineno: 22 doc: 'Create a binary classification sample dataset and save. If no filename is given it will default to: @@ -59,14 +60,13 @@ spec: Additional scikit-learn parameters can be set using **sk_params, please see https://scikit-learn.org/stable/modules/generated/sklearn.datasets.make_classification.html for more details.' - has_varargs: false name: gen_class_data - command: '' - disable_auto_mount: false - image: mlrun/mlrun build: origin_filename: '' functionSourceCode: IyBDb3B5cmlnaHQgMjAxOSBJZ3VhemlvCiMKIyBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKIyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCiMgWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0CiMKIyAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCiMKIyBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCiMgZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKIyBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KIyBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCiMgbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCiMKaW1wb3J0IHBhbmRhcyBhcyBwZApmcm9tIHR5cGluZyBpbXBvcnQgT3B0aW9uYWwsIExpc3QKZnJvbSBza2xlYXJuLmRhdGFzZXRzIGltcG9ydCBtYWtlX2NsYXNzaWZpY2F0aW9uCgpmcm9tIG1scnVuLmV4ZWN1dGlvbiBpbXBvcnQgTUxDbGllbnRDdHgKCgpkZWYgZ2VuX2NsYXNzX2RhdGEoCiAgICAgICAgY29udGV4dDogTUxDbGllbnRDdHgsCiAgICAgICAgbl9zYW1wbGVzOiBpbnQsCiAgICAgICAgbV9mZWF0dXJlczogaW50LAogICAgICAgIGtfY2xhc3NlczogaW50LAogICAgICAgIGhlYWRlcjogT3B0aW9uYWxbTGlzdFtzdHJdXSwKICAgICAgICBsYWJlbF9jb2x1bW46IE9wdGlvbmFsW3N0cl0gPSAibGFiZWxzIiwKICAgICAgICB3ZWlnaHQ6IGZsb2F0ID0gMC41LAogICAgICAgIHJhbmRvbV9zdGF0ZTogaW50ID0gMSwKICAgICAgICBrZXk6IHN0ciA9ICJjbGFzc2lmaWVyLWRhdGEiLAogICAgICAgIGZpbGVfZXh0OiBzdHIgPSAicGFycXVldCIsCiAgICAgICAgc2tfcGFyYW1zPXt9Cik6CiAgICAiIiJDcmVhdGUgYSBiaW5hcnkgY2xhc3NpZmljYXRpb24gc2FtcGxlIGRhdGFzZXQgYW5kIHNhdmUuCiAgICBJZiBubyBmaWxlbmFtZSBpcyBnaXZlbiBpdCB3aWxsIGRlZmF1bHQgdG86CiAgICAic2ltZGF0YS17bl9zYW1wbGVzfVh7bV9mZWF0dXJlc30ucGFycXVldCIuCgogICAgQWRkaXRpb25hbCBzY2lraXQtbGVhcm4gcGFyYW1ldGVycyBjYW4gYmUgc2V0IHVzaW5nICoqc2tfcGFyYW1zLCBwbGVhc2Ugc2VlIGh0dHBzOi8vc2Npa2l0LWxlYXJuLm9yZy9zdGFibGUvbW9kdWxlcy9nZW5lcmF0ZWQvc2tsZWFybi5kYXRhc2V0cy5tYWtlX2NsYXNzaWZpY2F0aW9uLmh0bWwgZm9yIG1vcmUgZGV0YWlscy4KCiAgICA6cGFyYW0gY29udGV4dDogICAgICAgZnVuY3Rpb24gY29udGV4dAogICAgOnBhcmFtIG5fc2FtcGxlczogICAgIG51bWJlciBvZiByb3dzL3NhbXBsZXMKICAgIDpwYXJhbSBtX2ZlYXR1cmVzOiAgICBudW1iZXIgb2YgY29scy9mZWF0dXJlcwogICAgOnBhcmFtIGtfY2xhc3NlczogICAgIG51bWJlciBvZiBjbGFzc2VzCiAgICA6cGFyYW0gaGVhZGVyOiAgICAgICAgaGVhZGVyIGZvciBmZWF0dXJlcyBhcnJheQogICAgOnBhcmFtIGxhYmVsX2NvbHVtbjogIGNvbHVtbiBuYW1lIG9mIGdyb3VuZC10cnV0aCBzZXJpZXMKICAgIDpwYXJhbSB3ZWlnaHQ6ICAgICAgICBmcmFjdGlvbiBvZiBzYW1wbGUgbmVnYXRpdmUgdmFsdWUgKGdyb3VuZC10cnV0aD0wKQogICAgOnBhcmFtIHJhbmRvbV9zdGF0ZTogIHJuZyBzZWVkIChzZWUgaHR0cHM6Ly9zY2lraXQtbGVhcm4ub3JnL3N0YWJsZS9nbG9zc2FyeS5odG1sI3Rlcm0tcmFuZG9tLXN0YXRlKQogICAgOnBhcmFtIGtleTogICAgICAgICAgIGtleSBvZiBkYXRhIGluIGFydGlmYWN0IHN0b3JlCiAgICA6cGFyYW0gZmlsZV9leHQ6ICAgICAgKHBxdCkgZXh0ZW5zaW9uIGZvciBwYXJxdWV0IGZpbGUKICAgIDpwYXJhbSBza19wYXJhbXM6ICAgICBhZGRpdGlvbmFsIHBhcmFtZXRlcnMgZm9yIGBza2xlYXJuLmRhdGFzZXRzLm1ha2VfY2xhc3NpZmljYXRpb25gCiAgICAiIiIKICAgIGZlYXR1cmVzLCBsYWJlbHMgPSBtYWtlX2NsYXNzaWZpY2F0aW9uKAogICAgICAgIG5fc2FtcGxlcz1uX3NhbXBsZXMsCiAgICAgICAgbl9mZWF0dXJlcz1tX2ZlYXR1cmVzLAogICAgICAgIHdlaWdodHM9d2VpZ2h0LAogICAgICAgIG5fY2xhc3Nlcz1rX2NsYXNzZXMsCiAgICAgICAgcmFuZG9tX3N0YXRlPXJhbmRvbV9zdGF0ZSwKICAgICAgICAqKnNrX3BhcmFtcykKCiAgICAjIG1ha2UgZGF0YWZyYW1lcywgYWRkIGNvbHVtbiBuYW1lcywgY29uY2F0ZW5hdGUgKFgsIHkpCiAgICBYID0gcGQuRGF0YUZyYW1lKGZlYXR1cmVzKQogICAgaWYgbm90IGhlYWRlcjoKICAgICAgICBYLmNvbHVtbnMgPSBbImZlYXRfIiArIHN0cih4KSBmb3IgeCBpbiByYW5nZShtX2ZlYXR1cmVzKV0KICAgIGVsc2U6CiAgICAgICAgWC5jb2x1bW5zID0gaGVhZGVyCgogICAgeSA9IHBkLkRhdGFGcmFtZShsYWJlbHMsIGNvbHVtbnM9W2xhYmVsX2NvbHVtbl0pCiAgICBkYXRhID0gcGQuY29uY2F0KFtYLCB5XSwgYXhpcz0xKQoKICAgIGNvbnRleHQubG9nX2RhdGFzZXQoa2V5LCBkZj1kYXRhLCBmb3JtYXQ9ZmlsZV9leHQsIGluZGV4PUZhbHNlKQo= code_origin: '' + command: '' + image: mlrun/mlrun + default_handler: gen_class_data + disable_auto_mount: false kind: job -verbose: false diff --git a/functions/src/gen_class_data/item.yaml b/functions/src/gen_class_data/item.yaml index 30f5cd21c..082b00305 100644 --- a/functions/src/gen_class_data/item.yaml +++ b/functions/src/gen_class_data/item.yaml @@ -11,7 +11,7 @@ labels: author: Iguazio maintainers: [] marketplaceType: '' -mlrunVersion: 1.7.0 +mlrunVersion: 1.10.0 name: gen_class_data platformVersion: 3.5.3 spec: @@ -21,4 +21,4 @@ spec: kind: job requirements: [] url: '' -version: 1.3.0 +version: 1.4.0 diff --git a/functions/src/gen_class_data/requirements.txt b/functions/src/gen_class_data/requirements.txt index d7dbe376b..e265290f6 100644 --- a/functions/src/gen_class_data/requirements.txt +++ b/functions/src/gen_class_data/requirements.txt @@ -1,2 +1,2 @@ pandas -scikit-learn==1.0.2 \ No newline at end of file +scikit-learn~=1.5.2 \ No newline at end of file diff --git a/functions/src/gen_class_data/test_gen_class_data.py b/functions/src/gen_class_data/test_gen_class_data.py index e06eeb16b..990075dec 100644 --- a/functions/src/gen_class_data/test_gen_class_data.py +++ b/functions/src/gen_class_data/test_gen_class_data.py @@ -36,4 +36,7 @@ def test_gen_class_data(): local=True, artifact_path="./artifacts", ) - assert os.path.isfile(run.status.artifacts[0]['spec']['target_path']), 'dataset is not available' + # In local mode, artifacts are in function-name/iteration subdirectory + # Default key is "classifier-data" (can be overridden in params) + dataset_path = "./artifacts/test-gen-class-data-gen-class-data/0/classifier-data.csv" + assert os.path.isfile(dataset_path), f'dataset is not available at {dataset_path}' From 04bf5087552f5db9b4bbbb030f4d895baff71a07 Mon Sep 17 00:00:00 2001 From: guy1992l <83535508+guy1992l@users.noreply.github.com> Date: Sun, 15 Feb 2026 15:22:14 +0200 Subject: [PATCH 2/3] [module][langchain_mlrun] First version (#963) * [Build] Fix html links, Add .html as source in documentation * Update CI temporarily and update index * [XGB-Custom] Fix test artifact key name * [XGB-Serving][XGB-Test][XGB-Trainer] Fix tests - artifact key * [Build] Install python 3.9 when testing (#618) * [Build] Update python version in CI (#620) * [Build] Install python 3.9 when testing * [Build] Update python version in CI * . * Revert "[Build] Update python version in CI (#620)" (#621) This reverts commit 0cd1f1585a618c253f201b6f5a63502cdbddb591. * Revert "[Build] Install python 3.9 when testing (#618)" (#619) This reverts commit 3301415200e52326bade1e17f99cb6b6d3880860. * [Build] Build with python 3.9 (#622) * [Build] Build with python 3.9 * . * [onnx utils] update onnx utils packages * [Noise-reduction] Add new function to hub (#765) * [Noise-reduction] Add new function to hub * fix test * added multiprocessing and silence removal to function * delete `load_dask` (#822) * [feature selection] update function yaml * [feature selection] update function yaml * Revert "[onnx utils] update onnx utils packages" This reverts commit 88727986ffa91662593958023be8ac3ccef2cab0. * [feature selection] update function yaml * [feature selection] update function yaml * Delete unsupported functions from the hub (#824) * delete EOS functions * bring back validate_great_expectations * bring back load_dataset * Update feature_selection/test_feature_selection.py Co-authored-by: Eyal Danieli * Update item.yaml * Align to master branch (#826) * [Category] Fix and add categories to functions (#808) * [Category] Fix and add categories to functions * bump version in structured * test is not valid in huggingface_serving * Fix duplicated footer * Fix duplicated footer * revert python version change as it will be done in another PR * comments * comments * Bump python:3.6 to python:3.9 (#810) * [Describe] Align describe to new pandas version (#812) * [Describe] Align describe to new pandas version * minor test fix * update mlrun version * add dask to requirements * remove dask * update numpy version * debug * debug * debug * remove dask tests * remove debug code * [get_offline_features] Updated to mlrun 1.6.3 (#813) * [Feature-selection] Replace matplotlib with plotly (#815) * Iguazio-cicd user token updated Iguazio-cicd user token updated in repo secrets: https://github.com/mlrun/functions/settings/secrets/actions MARKETPLACE_ACCESS_TOKEN_V3 new token gh...Zmf was set around April * forcing iguazio-cicd auth forcing iguazio-cicd to deal with Author identity unknown * checkout@v3 to v4 and echo * [Mlflow_utils] - mlflow model server (#811) * mlflow server * small fix to test * small fixes to ms and nb * small fixes to mlrun version * update requirements lightgbm * added req * Added xgboost to req --------- Co-authored-by: Avi Asulin <34214569+aviaIguazio@users.noreply.github.com> * [Mlflow] Remove mlflow tag (#825) * remove mlflow tag * remove mlflow tag --------- Co-authored-by: Avi Asulin <34214569+aviaIguazio@users.noreply.github.com> * align feature_selection yaml --------- Co-authored-by: Avi Asulin <34214569+aviaIguazio@users.noreply.github.com> Co-authored-by: Yonatan Shelach <92271540+yonishelach@users.noreply.github.com> Co-authored-by: rokatyy Co-authored-by: Katerina Molchanova <35141662+rokatyy@users.noreply.github.com> Co-authored-by: nashpaz123 <44337075+nashpaz123@users.noreply.github.com> Co-authored-by: ZeevRispler <73653682+ZeevRispler@users.noreply.github.com> * set `navigation_with_keys` to False (#829) * remove xgb and churn functions (#830) * [Batch Infer V2] Adjust function to 1.7 (#832) * adjust batch infer v2 * update docs in NB * bring back deprecated params and add warn (#834) * fix PyYAML loading (#837) * [text to audio generator] Replaced bark with openai tts models (#836) * [Text to audio generator] Add speech engine (#838) * [text to audio generator] Replaced bark with openai tts models * [text to audio generator] Fix base url env var * fix version * Add speech engine * after review * [auto-trainer] update test requirements (#839) * [Build] Fix html links, Add .html as source in documentation * Update CI temporarily and update index * [XGB-Custom] Fix test artifact key name * [XGB-Serving][XGB-Test][XGB-Trainer] Fix tests - artifact key * [Build] Install python 3.9 when testing (#618) * [Build] Update python version in CI (#620) * [Build] Install python 3.9 when testing * [Build] Update python version in CI * . * Revert "[Build] Update python version in CI (#620)" (#621) This reverts commit 0cd1f1585a618c253f201b6f5a63502cdbddb591. * Revert "[Build] Install python 3.9 when testing (#618)" (#619) This reverts commit 3301415200e52326bade1e17f99cb6b6d3880860. * [Build] Build with python 3.9 (#622) * [Build] Build with python 3.9 * . * Update requirements.txt * [Feature Selection] Fix deprecated `get_offline_features` (#844) * fix feature_selection * fix feature_selection * fix feature_selection nb * update yaml name * fix test * fix test * limit torchaudio for unit test (#845) * Update requirements.txt (#843) * [Open Archive] Fix arbitrary file vulnerability (#847) * fix arbitrary file vulnerability * fix arbitrary file vulnerability * fix test * [open_archive] Add traversal attack test (#849) * add traversal test * add traversal test * add traversal test * first version * run upg to pydantic v2 * added kafka and mlrun-ce code preparation * Eyal review * LangChain MLRun Integration with Kafka Support (CE Mode) (#1) * docstring syntax fixes * Add initial Kafka support for MLRun CE Implement _KafkaMLRunEndPointClient with KafkaProducer Add kafka_broker and kafka_topic to MLRunTracerClientSettings Add Kafka parameters to setup_langchain_monitoring() Update notebook to auto-detect CE/Enterprise mode Add kafka-python, orjson, uuid-utils to requirements.txt * added KAFKA_BROKER and DatastoreProfileTDEngine for registering kafka and tsdb profiles * clear output * use get_kafka_topic for the project name remove "raises:" docstring added kafka flush added s3fs to requirements.txt * adaptive notebook for local execution - AWS_ENDPOINT_URL_S3 env variable in deployment - port forwarding scripts * Replace kafka_broker/kafka_topic with stream_profile_name - Update _KafkaMLRunEndPointClient to use DatastoreProfileKafkaStream - Fetch Kafka config (broker, topic, SASL, SSL) from registered profile - Auto-retrieve stream_profile_name from model monitoring credentials - Update MLRunTracerClientSettings with new stream_profile_name field - Update setup_langchain_monitoring() to use profile-based config - Update notebook to use simplified API * Add get_kafka_stream_profile_name() utility to auto-detect Kafka profile * Move dependencies from requirements.txt to item.yaml * generalize env variables * Deploy monitoring app once instead of redeploy Unify profile variable naming for CE and Enterprise modes * _KafkaMLRunEndPointClient constructor - keep project handling NONE able with parent class handling in * revert removal of raise docstring enforce usage of stream_profile_name * change stream_profile_name to kafka_stream_profile_name revert ValueError message * Revert ValueError message Added mechanism to flush stream upon root run (instead of each monitor call) * ensure self._mlrun_client is not None before flush * revert double deploy * Add configurable kafka_linger_ms for hybrid flush control Adds kafka_linger_ms parameter to control message delivery timing: - Explicit flush mode (linger_ms=0, default): flush after each root run - Kafka-managed mode (linger_ms>0): Kafka controls delivery timing The flush() method now handles the mode internally - it's a no-op when Kafka-managed mode is enabled, keeping the tracer code simple. * fix double deploy in notebook (set the in-cluster MinIO endpoint before deploying) * Simplify Kafka flush behavior and enable batching by default - Always flush at end of root run (removed conditional linger_ms check) - Set default kafka_linger_ms to 500ms for message batching - Simplify KafkaProducer initialization (pop bootstrap_servers instead of lambda) * remove local module import from notebook remove redundant requirements from item.yaml * move env variables setup to top of the notebook * renamed env vars * fix test * gilad review --------- Co-authored-by: yonishelach Co-authored-by: Yoni Shelach <92271540+yonishelach@users.noreply.github.com> Co-authored-by: Avi Asulin Co-authored-by: Eyal Danieli Co-authored-by: Avi Asulin <34214569+aviaIguazio@users.noreply.github.com> Co-authored-by: rokatyy Co-authored-by: Katerina Molchanova <35141662+rokatyy@users.noreply.github.com> Co-authored-by: nashpaz123 <44337075+nashpaz123@users.noreply.github.com> Co-authored-by: ZeevRispler <73653682+ZeevRispler@users.noreply.github.com> Co-authored-by: daniels290813 <78727943+daniels290813@users.noreply.github.com> Co-authored-by: Tomer Weitzman <81749152+tomerbv@users.noreply.github.com> --- modules/src/langchain_mlrun/item.yaml | 24 + .../src/langchain_mlrun/langchain_mlrun.ipynb | 1046 ++++++++++ .../src/langchain_mlrun/langchain_mlrun.py | 1840 +++++++++++++++++ .../notebook_images/mlrun_ui.png | Bin 0 -> 85919 bytes modules/src/langchain_mlrun/requirements.txt | 4 + .../langchain_mlrun/test_langchain_mlrun.py | 1025 +++++++++ 6 files changed, 3939 insertions(+) create mode 100644 modules/src/langchain_mlrun/item.yaml create mode 100644 modules/src/langchain_mlrun/langchain_mlrun.ipynb create mode 100644 modules/src/langchain_mlrun/langchain_mlrun.py create mode 100644 modules/src/langchain_mlrun/notebook_images/mlrun_ui.png create mode 100644 modules/src/langchain_mlrun/requirements.txt create mode 100644 modules/src/langchain_mlrun/test_langchain_mlrun.py diff --git a/modules/src/langchain_mlrun/item.yaml b/modules/src/langchain_mlrun/item.yaml new file mode 100644 index 000000000..532cb4bd3 --- /dev/null +++ b/modules/src/langchain_mlrun/item.yaml @@ -0,0 +1,24 @@ +apiVersion: v1 +categories: +- langchain +- langgraph +- tracing +- monitoring +- llm +description: LangChain x MLRun integration - Orchestrate your LangChain code with MLRun. +example: langchain_mlrun.ipynb +generationDate: 2026-01-08:12-25 +hidden: false +labels: + author: Iguazio +mlrunVersion: 1.10.0 +name: langchain_mlrun +spec: + filename: langchain_mlrun.py + image: mlrun/mlrun + kind: generic + requirements: + - langchain~=1.2 + - pydantic-settings~=2.12 + - kafka-python~=2.3 +version: 0.0.1 \ No newline at end of file diff --git a/modules/src/langchain_mlrun/langchain_mlrun.ipynb b/modules/src/langchain_mlrun/langchain_mlrun.ipynb new file mode 100644 index 000000000..0e5a341e7 --- /dev/null +++ b/modules/src/langchain_mlrun/langchain_mlrun.ipynb @@ -0,0 +1,1046 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "7955da79-02cc-42fe-aee0-5456d3e386fd", + "metadata": {}, + "source": [ + "# LangChain ✕ MLRun Integration\n", + "\n", + "`langchain_mlrun` is a hub module that implements LangChain integration with MLRun. Using the module allows MLRun to orchestrate LangChain and LangGraph code, enabling tracing and monitoring batch workflows and realtime deployments.\n", + "___" + ] + }, + { + "cell_type": "markdown", + "id": "8392a3e1-d0a1-409a-ae68-fcc36858d30a", + "metadata": {}, + "source": [ + "## Main Components\n", + "\n", + "This is a short brief of the components available to import from the `langchain_mlrun` module. For full docs, see the documentation page.\n", + "\n", + "### Settings\n", + "\n", + "The module uses Pydantic settings classes that can be configured programmatically or via environment variables. The main class is `MLRunTracerSettings`. It contains two sub-settings:\n", + "* `MLRunTracerClientSettings` - Connection settings (stream path, container, endpoint info). Env prefix: `\"LC_MLRUN_TRACER_CLIENT_\"`\n", + "* `MLRunTracerMonitorSettings` - Controls what/how runs are captured (filters, labels, debug mode). Env prefix: `\"LC_MLRUN_TRACER_MONITOR_\"`\n", + "\n", + "For more information about each setting, see the class docstrings.\n", + "\n", + "#### Example - via code configuration\n", + "\n", + "```python\n", + "from langchain_mlrun import MLRunTracerSettings, MLRunTracerClientSettings, MLRunTracerMonitorSettings\n", + "\n", + "settings = MLRunTracerSettings(\n", + " client=MLRunTracerClientSettings(\n", + " stream_path=\"my-project/model-endpoints/stream-v1\",\n", + " container=\"projects\",\n", + " model_endpoint_name=\"my_endpoint\",\n", + " model_endpoint_uid=\"abc123\",\n", + " serving_function=\"my_function\",\n", + " ),\n", + " monitor=MLRunTracerMonitorSettings(\n", + " label=\"production\",\n", + " root_run_only=True, # Only monitor root runs, not child runs\n", + " tags_filter=[\"important\"], # Only monitor runs with this tag\n", + " ),\n", + ")\n", + "```\n", + "\n", + "#### Example - environment variable configuration\n", + "\n", + "```bash\n", + "export LC_MLRUN_TRACER_CLIENT_STREAM_PATH=\"my-project/model-endpoints/stream-v1\"\n", + "export LC_MLRUN_TRACER_CLIENT_CONTAINER=\"projects\"\n", + "export LC_MLRUN_TRACER_MONITOR_LABEL=\"production\"\n", + "export LC_MLRUN_TRACER_MONITOR_ROOT_RUN_ONLY=\"true\"\n", + "```\n", + "\n", + "### MLRun Tracer\n", + "\n", + "`MLRunTracer` is a LangChain-compatible tracer that converts LangChain `Run` objects into MLRun monitoring events and publishes them to a V3IO stream. \n", + "\n", + "Key points:\n", + "* **No inheritance required** - use it directly without subclassing.\n", + "* **Fully customizable via settings** - control filtering, summarization, and output format.\n", + "* **Custom summarizer support** - pass your own `run_summarizer_function` via settings to customize how runs are converted to events.\n", + "\n", + "### Monitoring Setup Utility Function\n", + "\n", + "`setup_langchain_monitoring()` is a utility function that creates the necessary MLRun infrastructure for LangChain monitoring. This is a **temporary workaround** until custom endpoint creation support is added to MLRun.\n", + "\n", + "The function returns a dictionary of environment variables to configure auto-tracing. See how to use it in the tutorial section below.\n", + "\n", + "### LangChain Monitoring Application\n", + "\n", + "`LangChainMonitoringApp` is a base class (inheriting from MLRun's `ModelMonitoringApplicationBase`) for building monitoring applications that process events from the MLRun Tracer.\n", + "\n", + "It offers several built-in helper methods and metrics for analyzing LangChain runs:\n", + "\n", + "* Helper methods:\n", + " * `get_structured_runs()` - Parse raw monitoring samples into structured run dictionaries with filtering options\n", + " * `iterate_structured_runs()` - Iterate over all runs including nested child runs\n", + "* Metric methods:\n", + " * `calculate_average_latency()` - Average latency across root runs\n", + " * `calculate_success_rate()` - Percentage of runs without errors\n", + " * `count_token_usage()` - Total input/output tokens from LLM runs\n", + " * `count_run_names()` - Count occurrences of each run name\n", + "\n", + "The base app can be used as-is, but it is recommended to extend it with your own custom monitoring logic.\n", + "___" + ] + }, + { + "cell_type": "markdown", + "id": "7e24e1a5-d80a-4b7e-9b94-57b24e8b39d7", + "metadata": {}, + "source": [ + "## How to Apply MLRun?\n", + "\n", + "### Auto Tracing\n", + "\n", + "Auto tracing automatically instruments all LangChain code by setting the `LC_MLRUN_MONITORING_ENABLED` environment variable and importing the module:\n", + "\n", + "```python\n", + "import os\n", + "os.environ[\"LC_MLRUN_MONITORING_ENABLED\"] = \"1\"\n", + "# Set other LC_MLRUN_TRACER_* environment variables as needed...\n", + "\n", + "# Import the module BEFORE any LangChain code\n", + "langchain_mlrun = mlrun.import_module(\"hub://langchain_mlrun\")\n", + "\n", + "# All LangChain/LangGraph code below will be automatically traced\n", + "chain.invoke(...)\n", + "```\n", + "\n", + "### Manual Tracing\n", + "\n", + "For more control, use the `mlrun_monitoring()` context manager to trace specific code blocks:\n", + "\n", + "```python\n", + "langchain_mlrun = mlrun.import_module(\"hub://langchain_mlrun\")\n", + "mlrun_monitoring = langchain_mlrun.mlrun_monitoring\n", + "MLRunTracerSettings = langchain_mlrun.MLRunTracerSettings\n", + "\n", + "# Optional: customize settings\n", + "settings = MLRunTracerSettings(...)\n", + "\n", + "with mlrun_monitoring(settings=settings) as tracer:\n", + " # Only LangChain code within this block will be traced\n", + " result = chain.invoke({\"topic\": \"MLRun\"})\n", + "```\n", + "___" + ] + }, + { + "cell_type": "markdown", + "id": "68b52d3d-a431-44fb-acd6-ea33fec37a49", + "metadata": {}, + "source": [ + "## Tutorial\n", + "\n", + "In this tutorial we'll show how to orchestrate LangChain based code with MLRun using the `langchain_mlrun` hub module.\n", + "\n", + "### Prerequisites\n", + "\n", + "Install MLRun and the `langchain_mlrun` requirements." + ] + }, + { + "cell_type": "code", + "id": "caf72aa6-06e8-4a04-bfc4-409b39d255fe", + "metadata": {}, + "source": "!pip install mlrun langchain~=1.2 pydantic-settings~=2.12 kafka-python~=2.3", + "outputs": [], + "execution_count": null + }, + { + "cell_type": "markdown", + "id": "vprq8wj4iqh", + "source": [ + "### Local Development Setup (Optional)\n", + "\n", + "> Skip this section if you're running inside a Jupyter instance deployed in the MLRun cluster.\n", + "\n", + "If you're running this notebook from your local machine, follow these steps:\n", + "\n", + "#### Step 1: Set Environment Variables\n", + "\n", + "Run the cell below to set up all required environment variables for local development." + ], + "metadata": {} + }, + { + "cell_type": "code", + "id": "9lc788zu3zi", + "source": [ + "import os\n", + "\n", + "# MLRun API endpoint:\n", + "# os.environ[\"MLRUN_DBPATH\"] = \"http://localhost:30070\"\n", + "\n", + "# Kafka Configuration:\n", + "# os.environ[\"KAFKA_BROKER\"] = \"\"\n", + "\n", + "# TDEngine Configuration:\n", + "# os.environ[\"TDENGINE_HOST\"] = \"\"\n", + "# os.environ[\"TDENGINE_PORT\"] = \"\"\n", + "# os.environ[\"TDENGINE_USER\"] = \"\"\n", + "# os.environ[\"TDENGINE_PASSWORD\"] = \"\"\n", + "\n", + "# MinIO/S3 Configuration:\n", + "# os.environ[\"AWS_ACCESS_KEY_ID\"] = \"\"\n", + "# os.environ[\"AWS_SECRET_ACCESS_KEY\"] = \"\"\n", + "# os.environ[\"AWS_ENDPOINT_URL_S3\"] = \"\"" + ], + "metadata": {}, + "execution_count": null, + "outputs": [] + }, + { + "metadata": {}, + "cell_type": "markdown", + "source": [ + "#### Step 2: Set Up Port Forwarding\n", + "\n", + "Set up port-forwarding to access cluster services. Run these commands in separate terminal windows:\n", + "\n", + "```bash\n", + "# MLRun API\n", + "kubectl port-forward -n mlrun svc/mlrun-api 30070:8080\n", + "```\n", + "\n", + "```bash\n", + "# MinIO (S3-compatible storage)\n", + "kubectl port-forward -n mlrun svc/minio 9000:9000\n", + "```\n", + "\n", + "```bash\n", + "# Kafka (for CE mode) - requires /etc/hosts entry: 127.0.0.1 kafka-stream\n", + "kubectl port-forward -n mlrun svc/kafka-stream 9092:9092\n", + "```\n", + "\n", + "```bash\n", + "# TDEngine (for CE mode) - requires /etc/hosts entry: 127.0.0.1 tdengine-tsdb\n", + "kubectl port-forward -n mlrun svc/tdengine-tsdb 6041:6041\n", + "```" + ], + "id": "6d1d2d3c016ec62c" + }, + { + "metadata": {}, + "cell_type": "markdown", + "source": [ + "### Create Project\n", + "\n", + "We'll first create an MLRun project" + ], + "id": "4442f7ad1b0a8ee" + }, + { + "cell_type": "code", + "id": "2664df3e-d9c6-40dd-a215-29d60e4b4208", + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-03T19:43:18.142870Z", + "start_time": "2026-02-03T19:43:10.068758Z" + } + }, + "source": [ + "import time\n", + "import datetime\n", + "import mlrun\n", + "\n", + "print(f\"MLRun version: {mlrun.__version__}\")\n", + "print(f\"CE Mode: {mlrun.mlconf.is_ce_mode()}\")\n", + "\n", + "project = mlrun.get_or_create_project(\"langchain-mlrun-tutorial\")\n", + "print(f\"Project: {project.name}\")" + ], + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "MLRun version: 1.10.0\n", + "CE Mode: True\n", + "> 2026-02-03 21:43:18,053 [info] Loading project from path: {\"path\":\"./\",\"project_name\":\"langchain-mlrun-tutorial\",\"user_project\":false}\n", + "> 2026-02-03 21:43:18,141 [info] Project loaded successfully: {\"path\":\"./\",\"project_name\":\"langchain-mlrun-tutorial\",\"stored_in_db\":true}\n", + "Project: langchain-mlrun-tutorial\n" + ] + } + ], + "execution_count": 3 + }, + { + "cell_type": "markdown", + "id": "33f28986-c158-47fd-97a6-74f69892b4eb", + "metadata": {}, + "source": "### Enable Monitoring\n\nTo use MLRun's monitoring feature in our project we first need to set up the monitoring infrastructure.\n\n- **MLRun CE**: Uses Kafka for streaming (automatically detected)\n- **MLRun Enterprise**: Uses V3IO for streaming (automatically detected)\n\nThe cell below automatically detects your MLRun mode and sets up the appropriate streaming infrastructure." + }, + { + "cell_type": "code", + "id": "d9d2fa66-0498-445d-ab4a-8370f46aec1e", + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-03T19:49:22.700332Z", + "start_time": "2026-02-03T19:43:18.148037Z" + } + }, + "source": [ + "# Create datastore profiles (based on CE or Enterprise):\n", + "if mlrun.mlconf.is_ce_mode():\n", + " print(\"Setting up Kafka streaming for MLRun CE...\")\n", + " from mlrun.datastore.datastore_profile import DatastoreProfileKafkaStream, DatastoreProfileTDEngine\n", + " \n", + " stream_profile = DatastoreProfileKafkaStream(\n", + " name=\"kafka-stream-profile\",\n", + " brokers=os.environ[\"KAFKA_BROKER\"],\n", + " topics=[],\n", + " )\n", + " tsdb_profile = DatastoreProfileTDEngine(\n", + " name=\"tsdb-profile\",\n", + " user=os.environ[\"TDENGINE_USER\"],\n", + " password=os.environ[\"TDENGINE_PASSWORD\"],\n", + " host=os.environ[\"TDENGINE_HOST\"],\n", + " port=int(os.environ[\"TDENGINE_PORT\"]),\n", + " )\n", + " project.register_datastore_profile(stream_profile)\n", + " project.register_datastore_profile(tsdb_profile)\n", + "else: # Enterprise\n", + " print(\"Setting up V3IO streaming for MLRun Enterprise...\")\n", + " from mlrun.datastore import DatastoreProfileV3io\n", + " \n", + " stream_profile = DatastoreProfileV3io(name=\"v3io-ds\", v3io_access_key=os.environ[\"V3IO_ACCESS_KEY\"])\n", + " tsdb_profile = stream_profile\n", + " project.register_datastore_profile(stream_profile)\n", + "\n", + "# Enable monitoring in our project:\n", + "project.set_model_monitoring_credentials(\n", + " stream_profile_name=stream_profile.name,\n", + " tsdb_profile_name=tsdb_profile.name,\n", + ")\n", + "project.enable_model_monitoring(\n", + " base_period=1,\n", + " wait_for_deployment=True,\n", + ")\n", + "\n", + "print(\"Monitoring enabled successfully!\")" + ], + "outputs": [], + "execution_count": null + }, + { + "cell_type": "markdown", + "id": "f23117fa-7b67-470c-80ca-976d14c2120e", + "metadata": {}, + "source": [ + "### Import `langchain_mlrun`\n", + "\n", + "Now we'll import `langchain_mlrun` from the hub." + ] + }, + { + "cell_type": "code", + "id": "2360cd49-b260-4140-bd16-138349e000b3", + "metadata": {}, + "source": [ + "# Import the module from the hub:\n", + "langchain_mlrun = mlrun.import_module(\"hub://langchain_mlrun\")\n", + "\n", + "# Import the utility function and monitoring application from the module:\n", + "setup_langchain_monitoring = langchain_mlrun.setup_langchain_monitoring\n", + "LangChainMonitoringApp = langchain_mlrun.LangChainMonitoringApp" + ], + "outputs": [], + "execution_count": null + }, + { + "cell_type": "markdown", + "id": "de030131-ebaf-48f8-96ed-3c1013b5e260", + "metadata": {}, + "source": [ + "### Create Monitorable Endpoint\n", + "\n", + "Endpoints are the entities being monitored by MLRun. We'll use the `setup_langchain_monitoring()` utility function to create the model monitoring endpoint.\n", + "\n", + "For MLRun CE mode, you must pass the `kafka_stream_profile_name` parameter with the name of the registered Kafka stream profile.\n", + "\n", + "By default, the endpoint name will be `\"langchain_mlrun_endpoint\"` but you can change it by using the `model_endpoint_name` parameter." + ] + }, + { + "cell_type": "code", + "id": "0e9baf78-3d38-46bd-89dd-6f83760eaeb0", + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-03T19:49:23.861085Z", + "start_time": "2026-02-03T19:49:23.412235Z" + } + }, + "source": [ + "# Pass kafka_stream_profile_name for CE mode (required)\n", + "env_vars = setup_langchain_monitoring(\n", + " kafka_stream_profile_name=stream_profile.name if mlrun.mlconf.is_ce_mode() else None\n", + ")" + ], + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Creating LangChain model endpoint\n", + "\n", + " [✓] Loading Project......................... Done (0.00s)\u001B[K\n", + " [✓] Creating Model.......................... Done (0.31s) \u001B[K\n", + " [✓] Creating Function....................... Done (0.04s) \u001B[K\n", + " [✓] Creating Model Endpoint................. Done (0.09s) \u001B[K\n", + "\n", + "✨ Done! LangChain monitoring model endpoint created successfully.\n", + "You can now set the following environment variables to enable MLRun tracing in your LangChain code:\n", + "\n", + "{\n", + " \"MLRUN_MONITORING_ENABLED\": \"1\",\n", + " \"MLRUN_TRACER_CLIENT_PROJECT\": \"langchain-mlrun-tutorial\",\n", + " \"MLRUN_TRACER_CLIENT_MODEL_ENDPOINT_NAME\": \"langchain_mlrun_endpoint\",\n", + " \"MLRUN_TRACER_CLIENT_MODEL_ENDPOINT_UID\": \"d1d2b2686772441cacf687b45cd48ffa\",\n", + " \"MLRUN_TRACER_CLIENT_SERVING_FUNCTION\": \"langchain_mlrun_function\",\n", + " \"MLRUN_TRACER_CLIENT_KAFKA_STREAM_PROFILE_NAME\": \"kafka-stream-profile\"\n", + "}\n", + "\n", + "To customize the monitoring behavior, you can also set additional environment variables prefixed with 'MLRUN_TRACER_MONITOR_'. Refer to the MLRun tracer documentation for more details.\n", + "\n" + ] + } + ], + "execution_count": 6 + }, + { + "cell_type": "markdown", + "id": "dd45c94b-ee05-449c-9336-0aa659e66bda", + "metadata": {}, + "source": [ + "### Setup Environment Variables for Auto Tracing\n", + "\n", + "We'll use the environment variables returned from `setup_langchain_monitoring` to setup the environment for auto-tracing. Read the printed outputs for more information." + ] + }, + { + "cell_type": "code", + "id": "1c1988f8-c80a-4bf2-bfb1-d43523fc161f", + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-03T19:49:23.866556Z", + "start_time": "2026-02-03T19:49:23.864805Z" + } + }, + "source": [ + "os.environ.update(env_vars)" + ], + "outputs": [], + "execution_count": 7 + }, + { + "cell_type": "markdown", + "id": "d3f3b8e5-3538-4153-95da-e6d8776be3ac", + "metadata": {}, + "source": "### Run `langchain` or `langgraph` Code\n\nHere we have 3 functions, each using different method utilizing LLMs with `langchain` and `langgraph`:\n* `run_simple_chain` - Using `langchain`'s chains.\n* `run_simple_agent` - Using `langchain`'s `create_agent` function and `tool`s.\n* `run_langgraph_graph` - Using pure `langgraph`.\n\n> **Notice**: You don't need to set OpenAI API credentials, there is a mock `ChatModel` that will replace it if the credentials are not set in the environment. If you wish to use OpenAI models, make sure you `pip install langchain_openai` and set the `OPENAI_API_KEY` environment variable before continue to the next cell.\n\nBecause the auto-tracing environment is set, any run will be automatically traced and monitored!\n\nFeel free to adjust the code as you like.\n\n> **Remember**: To enable auto-tracing you do need to set the environment variables and import the `langchain_mlrun` module before any LangChain code. For batch jobs and realtime functions, make sure you set env vars in the MLRun function and add the import line `langchain_mlrun = mlrun.import_module(\"hub://langchain_mlrun\")` at the top of your code." + }, + { + "cell_type": "code", + "id": "94b4d4b0-8d10-4ad3-8f16-7b1b7daeac11", + "metadata": { + "tags": [], + "ExecuteTime": { + "end_time": "2026-02-03T19:49:24.899991Z", + "start_time": "2026-02-03T19:49:23.869475Z" + } + }, + "source": [ + "import os\n", + "from typing import Literal, TypedDict, Annotated, Sequence, Any, Callable\n", + "from operator import add\n", + "\n", + "from langchain_core.language_models import LanguageModelInput\n", + "from langchain_core.runnables import Runnable, RunnableLambda\n", + "from langchain_core.prompts import ChatPromptTemplate\n", + "from langchain_core.output_parsers import StrOutputParser\n", + "from langchain_core.language_models.fake_chat_models import FakeListChatModel, GenericFakeChatModel\n", + "from langchain.agents import create_agent\n", + "from langchain_core.messages import AIMessage, HumanMessage\n", + "from langchain_core.tools import tool, BaseTool\n", + "\n", + "from langgraph.graph import StateGraph, START, END\n", + "from langchain_core.messages import BaseMessage\n", + "\n", + "\n", + "def _check_openai_credentials() -> bool:\n", + " \"\"\"\n", + " Check if OpenAI API key is set in environment variables.\n", + "\n", + " :return: True if OPENAI_API_KEY is set, False otherwise.\n", + " \"\"\"\n", + " return \"OPENAI_API_KEY\" in os.environ\n", + "\n", + "\n", + "# Import ChatOpenAI only if OpenAI credentials are available (meaning `langchain-openai` must be installed).\n", + "if _check_openai_credentials():\n", + " from langchain_openai import ChatOpenAI\n", + "\n", + " \n", + "class _ToolEnabledFakeModel(GenericFakeChatModel):\n", + " \"\"\"\n", + " A fake chat model that supports tool binding for running agent tracing tests.\n", + " \"\"\"\n", + "\n", + " def bind_tools(\n", + " self,\n", + " tools: Sequence[\n", + " dict[str, Any] | type | Callable | BaseTool # noqa: UP006\n", + " ],\n", + " *,\n", + " tool_choice: str | None = None,\n", + " **kwargs: Any,\n", + " ) -> Runnable[LanguageModelInput, AIMessage]:\n", + " return self\n", + "\n", + "\n", + "#: Tag value for testing tag filtering.\n", + "_dummy_tag = \"dummy_tag\"\n", + "\n", + "\n", + "def run_simple_chain() -> str:\n", + " \"\"\"\n", + " Run a simple LangChain chain that gets a fact about a topic.\n", + " \"\"\"\n", + " # Build a simple chain: prompt -> llm -> str output parser\n", + " llm = ChatOpenAI(\n", + " model=\"gpt-4o-mini\",\n", + " tags=[_dummy_tag]\n", + " ) if _check_openai_credentials() else (\n", + " FakeListChatModel(\n", + " responses=[\n", + " \"MLRun is an open-source orchestrator for machine learning pipelines.\"\n", + " ],\n", + " tags=[_dummy_tag]\n", + " )\n", + " )\n", + " prompt = ChatPromptTemplate.from_template(\"Tell me a short fact about {topic}\")\n", + " chain = prompt | llm | StrOutputParser()\n", + "\n", + " # Run the chain:\n", + " response = chain.invoke({\"topic\": \"MLRun\"})\n", + " return response\n", + "\n", + "\n", + "def run_simple_agent():\n", + " \"\"\"\n", + " Run a simple LangChain agent that uses two tools to get weather and stock price.\n", + " \"\"\"\n", + " # Define the tools:\n", + " @tool\n", + " def get_weather(city: str) -> str:\n", + " \"\"\"Get the current weather for a specific city.\"\"\"\n", + " return f\"The weather in {city} is 22°C and sunny.\"\n", + "\n", + " @tool\n", + " def get_stock_price(symbol: str) -> str:\n", + " \"\"\"Get the current stock price for a symbol.\"\"\"\n", + " return f\"The stock price for {symbol} is $150.25.\"\n", + "\n", + " # Define the model:\n", + " model = ChatOpenAI(\n", + " model=\"gpt-4o-mini\",\n", + " tags=[_dummy_tag]\n", + " ) if _check_openai_credentials() else (\n", + " _ToolEnabledFakeModel(\n", + " messages=iter(\n", + " [\n", + " AIMessage(\n", + " content=\"\",\n", + " tool_calls=[\n", + " {\"name\": \"get_weather\", \"args\": {\"city\": \"London\"}, \"id\": \"call_abc123\"},\n", + " {\"name\": \"get_stock_price\", \"args\": {\"symbol\": \"AAPL\"}, \"id\": \"call_def456\"}\n", + " ]\n", + " ),\n", + " AIMessage(content=\"The weather in London is 22°C and AAPL is trading at $150.25.\")\n", + " ]\n", + " ),\n", + " tags=[_dummy_tag]\n", + " )\n", + " )\n", + "\n", + " # Create the agent:\n", + " agent = create_agent(\n", + " model=model,\n", + " tools=[get_weather, get_stock_price],\n", + " system_prompt=\"You are a helpful assistant with access to tools.\"\n", + " )\n", + "\n", + " # Run the agent:\n", + " return agent.invoke({\"messages\": [\"What is the weather in London and the stock price of AAPL?\"]})\n", + "\n", + "\n", + "def run_langgraph_graph():\n", + " \"\"\"\n", + " Run a LangGraph agent that uses reflection to correct its answer.\n", + " \"\"\"\n", + " # Define the graph state:\n", + " class AgentState(TypedDict):\n", + " messages: Annotated[list[BaseMessage], add]\n", + " attempts: int\n", + "\n", + " # Define the model:\n", + " model = ChatOpenAI(model=\"gpt-4o-mini\") if _check_openai_credentials() else (\n", + " _ToolEnabledFakeModel(\n", + " messages=iter(\n", + " [\n", + " AIMessage(content=\"There are 2 'r's in Strawberry.\"), # Mocking the failure\n", + " AIMessage(content=\"I stand corrected. S-t-r-a-w-b-e-r-r-y. There are 3 'r's.\"), # Mocking the fix\n", + " ]\n", + " )\n", + " )\n", + " )\n", + "\n", + " # Define the graph nodes and router:\n", + " def call_model(state: AgentState):\n", + " response = model.invoke(state[\"messages\"])\n", + " return {\"messages\": [response], \"attempts\": state[\"attempts\"] + 1}\n", + "\n", + " def reflect_node(state: AgentState):\n", + " prompt = \"Wait, count the 'r's again slowly, letter by letter. Are you sure?\"\n", + " return {\"messages\": [HumanMessage(content=prompt)]}\n", + "\n", + " def router(state: AgentState) -> Literal[\"reflect\", END]:\n", + " # Make sure there are 2 attempts at least for an answer:\n", + " if state[\"attempts\"] == 1:\n", + " return \"reflect\"\n", + " return END\n", + "\n", + " # Build the graph:\n", + " builder = StateGraph(AgentState)\n", + " builder.add_node(\"model\", call_model)\n", + " tagged_reflect_node = RunnableLambda(reflect_node).with_config(tags=[_dummy_tag])\n", + " builder.add_node(\"reflect\", tagged_reflect_node)\n", + " builder.add_edge(START, \"model\")\n", + " builder.add_conditional_edges(\"model\", router)\n", + " builder.add_edge(\"reflect\", \"model\")\n", + " graph = builder.compile()\n", + "\n", + " # Run the graph:\n", + " return graph.invoke({\"messages\": [HumanMessage(content=\"How many 'r's in Strawberry?\")], \"attempts\": 0})" + ], + "outputs": [], + "execution_count": 8 + }, + { + "cell_type": "markdown", + "id": "49964f96-89ba-4f61-8788-38290a877aa2", + "metadata": {}, + "source": "Let's create some traffic, we'll run whatever function you want in a loop to get some events. We take timestamps in order to use them later to run the monitoring application on the data we'll send." + }, + { + "cell_type": "code", + "id": "b7e6418d-76f4-4b18-9ef9-c5bb40b20545", + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-03T22:05:54.601563Z", + "start_time": "2026-02-03T22:05:52.518385Z" + } + }, + "source": [ + "# Run LangChain code and now it should be tracked and monitored in MLRun:\n", + "start_timestamp = datetime.datetime.now() - datetime.timedelta(minutes=1)\n", + "for i in range(20):\n", + " run_simple_agent()\n", + "end_timestamp = datetime.datetime.now() + datetime.timedelta(minutes=5)" + ], + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-04 00:05:52,553 [info] Project loaded successfully: {\"project_name\":\"langchain-mlrun-tutorial\"}\n" + ] + } + ], + "execution_count": 13 + }, + { + "cell_type": "markdown", + "id": "d9085765-91fd-4d31-84b4-927ecf9cc455", + "metadata": {}, + "source": "> **Note**: Please wait a minute or two until the events are processed." + }, + { + "cell_type": "code", + "id": "85fae3e4-5f1b-4f0c-ba71-81060f10804f", + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-03T19:50:26.655189Z", + "start_time": "2026-02-03T19:49:26.648461Z" + } + }, + "source": [ + "time.sleep(60)" + ], + "outputs": [], + "execution_count": 10 + }, + { + "cell_type": "markdown", + "id": "2475ebec-fc32-4884-9723-3ca9cfde577f", + "metadata": {}, + "source": [ + "### Test the LangChain Monitoring Application\n", + "\n", + "To test a monitoring application, we use the `evaluate` class method. We'll run an evaluation on the data we just sent. It is a small local job and should run fast.\n", + "\n", + "Keep an eye for the returned metrics from the monitoring application." + ] + }, + { + "cell_type": "code", + "id": "3d046755-9153-497a-a024-5d63316e1f91", + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-03T19:50:28.003195Z", + "start_time": "2026-02-03T19:50:26.670024Z" + } + }, + "source": [ + "LangChainMonitoringApp.evaluate(\n", + " func_name=\"langchain-monitoring-app-test\",\n", + " func_path=\"langchain_mlrun.py\",\n", + " run_local=True,\n", + " endpoints=[env_vars[\"LC_MLRUN_TRACER_CLIENT_MODEL_ENDPOINT_NAME\"]],\n", + " start=start_timestamp.isoformat(),\n", + " end=end_timestamp.isoformat(),\n", + ")" + ], + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-03 21:50:26,671 [info] Changing function name - adding `\"-batch\"` suffix: {\"func_name\":\"langchain-monitoring-app-test-batch\"}\n", + "> 2026-02-03 21:50:26,815 [warning] It is recommended to use k8s secret (specify secret_name), specifying aws_access_key/aws_secret_key directly is unsafe.\n", + "> 2026-02-03 21:50:26,829 [info] Storing function: {\"db\":\"http://localhost:30070\",\"name\":\"langchain-monitoring-app-test-batch--handler\",\"uid\":\"f2c3c94681094915beb2c5c1ccc0dac8\"}\n", + "> 2026-02-03 21:50:27,953 [warning] No data was found for any of the specified endpoints. No results were produced: {\"application_name\":\"langchain-monitoring-app-test-batch\",\"end\":\"2026-02-03T21:54:26.640556\",\"endpoints\":[\"langchain_mlrun_endpoint\"],\"start\":\"2026-02-03T21:48:24.904667\"}\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ], + "text/html": [ + "\n", + "
\n", + "
\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
projectuiditerstartendstatekindnamelabelsinputsparametersresults
langchain-mlrun-tutorial
...c0dac8
0Feb 03 19:50:26NaTcompletedrunlangchain-monitoring-app-test-batch--handler
kind=local
owner=Tomer_Weitzman
host=M-QXN63PHMF9
endpoints=['langchain_mlrun_endpoint']
start=2026-02-03T21:48:24.904667
end=2026-02-03T21:54:26.640556
base_period=None
write_output=False
existing_data_handling=fail_on_overlap
stream_profile=None
\n", + "
\n", + "
\n", + "
\n", + " Title\n", + " ×\n", + "
\n", + " \n", + "
\n", + "
\n" + ] + }, + "metadata": {}, + "output_type": "display_data", + "jetTransient": { + "display_id": null + } + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ], + "text/html": [ + " > to track results use the .show() or .logs() methods " + ] + }, + "metadata": {}, + "output_type": "display_data", + "jetTransient": { + "display_id": null + } + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-03 21:50:28,001 [info] Run execution finished: {\"name\":\"langchain-monitoring-app-test-batch--handler\",\"status\":\"completed\"}\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ] + }, + "execution_count": 11, + "metadata": {}, + "output_type": "execute_result" + } + ], + "execution_count": 11 + }, + { + "cell_type": "markdown", + "id": "eda724c3-27f3-4d28-a7ba-1e59b9be2a37", + "metadata": {}, + "source": "### Deploy the Monitoring Application\n\nAll that's left to do now is to deploy our monitoring application!" + }, + { + "cell_type": "code", + "id": "652b00d4-070d-4849-9784-4d461cb83eae", + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-03T19:52:29.502406Z", + "start_time": "2026-02-03T19:50:28.009318Z" + } + }, + "source": "# Deploy the monitoring app:\nLangChainMonitoringApp.deploy(\n func_name=\"langchain-monitoring-app\",\n func_path=\"langchain_mlrun.py\",\n image=\"mlrun/mlrun\",\n requirements=[\n \"langchain\",\n \"pydantic-settings\",\n ],\n)", + "outputs": [], + "execution_count": null + }, + { + "cell_type": "markdown", + "id": "c23bef7a-cbdb-4b22-a2d9-2edbfde5eb04", + "metadata": {}, + "source": [ + "Once it is deployed, you can run events again and see the monitoring application in MLRun UI in action:\n", + "\n", + "![mlrun ui example](./notebook_images/mlrun_ui.png)" + ] + }, + { + "metadata": {}, + "cell_type": "code", + "outputs": [], + "execution_count": null, + "source": "", + "id": "fc994d2114a89a25" + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.14" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/modules/src/langchain_mlrun/langchain_mlrun.py b/modules/src/langchain_mlrun/langchain_mlrun.py new file mode 100644 index 000000000..920354bfb --- /dev/null +++ b/modules/src/langchain_mlrun/langchain_mlrun.py @@ -0,0 +1,1840 @@ +# Copyright 2026 Iguazio +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +""" +MLRun to LangChain integration - a tracer that converts LangChain Run objects into serializable event and send them to +MLRun monitoring. +""" + +from abc import ABC, abstractmethod +import copy +import importlib +import orjson +import os +import socket +from uuid import UUID +import threading +from contextlib import contextmanager +from contextvars import ContextVar +import datetime +from typing import Any, Callable, Generator, Optional + +from langchain_core.tracers import BaseTracer, Run +from langchain_core.tracers.context import register_configure_hook + +from pydantic import Field, field_validator, model_validator +from pydantic_settings import BaseSettings, SettingsConfigDict +from uuid_utils import uuid7 + +import mlrun +from mlrun.runtimes import RemoteRuntime +from mlrun.model_monitoring.applications import ( + ModelMonitoringApplicationBase, ModelMonitoringApplicationMetric, + ModelMonitoringApplicationResult, MonitoringApplicationContext, +) +import mlrun.common.schemas.model_monitoring.constants as mm_constants + +#: Environment variable name to use MLRun monitoring tracer via LangChain global tracing system: +mlrun_monitoring_env_var = "LC_MLRUN_MONITORING_ENABLED" + + +class _MLRunEndPointClient(ABC): + """ + An MLRun model endpoint monitoring client base class to connect and send events on a monitoring stream. + """ + + def __init__( + self, + model_endpoint_name: str, + model_endpoint_uid: str, + serving_function: str | RemoteRuntime, + serving_function_tag: str | None = None, + project: str | mlrun.projects.MlrunProject = None, + ): + """ + Initialize an MLRun model endpoint monitoring client. + + :param model_endpoint_name: The monitoring endpoint related model name. + :param model_endpoint_uid: Model endpoint unique identifier. + :param serving_function: Serving function name or ``RemoteRuntime`` object. + :param serving_function_tag: Optional function tag (defaults to 'latest'). + :param project: Project name or ``MlrunProject``. If ``None``, uses the current project. + raise: MLRunInvalidArgumentError: If there is no current active project and no `project` argument was provided. + """ + # Store the provided info: + self._model_endpoint_name = model_endpoint_name + self._model_endpoint_uid = model_endpoint_uid + + # Load project: + if project is None: + try: + self._project_name = mlrun.get_current_project(silent=False).name + except mlrun.errors.MLRunInvalidArgumentError: + raise mlrun.errors.MLRunInvalidArgumentError( + "There is no current active project. Either use `mlrun.get_or_create_project` prior to " + "initializing the monitoring tracer or pass a project name to load. You can also set the " + "environment variable: 'LC_MLRUN_TRACER_CLIENT_PROJECT'." + ) + elif isinstance(project, str): + self._project_name = project + else: + self._project_name = project.name + + # Load function: + if isinstance(serving_function, str): + self._serving_function_name = serving_function + self._serving_function_tag = serving_function_tag or "latest" + else: + self._serving_function_name = serving_function.metadata.name + self._serving_function_tag = ( + serving_function_tag or serving_function.metadata.tag + ) + + # Prepare the sample: + self._event_sample = { + "class": "CustomStream", + "worker": "0", + "model": self._model_endpoint_name, + "host": socket.gethostname(), + "function_uri": f"{self._project_name}/{self._serving_function_name}:{self._serving_function_tag}", + "endpoint_id": self._model_endpoint_uid, + "sampling_percentage": 100, + "request": {"inputs": [], "background_task_state": "succeeded"}, + "op": "infer", + "resp": { + "id": None, + "model_name": self._model_endpoint_name, + "outputs": [], + "timestamp": None, + "model_endpoint_uid": self._model_endpoint_uid, + }, + "when": None, + "microsec": 496, + "effective_sample_count": 1, + } + + @abstractmethod + def monitor( + self, + event_id: str, + label: str, + input_data: dict, + output_data: dict, + request_timestamp: str, + response_timestamp: str, + ): + """ + Monitor the provided event, sending it to the model endpoint monitoring stream. + + :param event_id: Unique event identifier used as the monitored record id. + :param label: Label for the run/event. + :param input_data: Serialized input data for the run. + :param output_data: Serialized output data for the run. + :param request_timestamp: Request/start timestamp in the format of '%Y-%m-%d %H:%M:%S%z'. + :param response_timestamp: Response/end timestamp in the format of '%Y-%m-%d %H:%M:%S%z'. + """ + pass + + def flush(self): + """ + Flush any buffered messages to ensure they are sent to the stream. + + For streaming backends that buffer messages (like Kafka), this ensures delivery. For backends that send + immediately (like V3IO), this may be a no-op. + """ + pass + + def _create_event( + self, + event_id: str, + label: str, + input_data: dict, + output_data: dict, + request_timestamp: str, + response_timestamp: str, + ) -> dict: + """ + Create a new event out of the stored event sample. + + :param event_id: Unique event identifier used as the monitored record id. + :param label: Label for the run/event. + :param input_data: Serialized input data for the run. + :param output_data: Serialized output data for the run. + :param request_timestamp: Request/start timestamp in the format of '%Y-%m-%d %H:%M:%S%z'. + :param response_timestamp: Response/end timestamp in the format of '%Y-%m-%d %H:%M:%S%z'. + + :returns: The event to send to the monitoring stream. + """ + # Copy the sample: + event = copy.deepcopy(self._event_sample) + + # Edit event with given parameters: + event["when"] = request_timestamp + event["request"]["inputs"].append(orjson.dumps({"label": label, "input": input_data}).decode('utf-8')) + event["resp"]["timestamp"] = response_timestamp + event["resp"]["outputs"].append(orjson.dumps(output_data).decode('utf-8')) + event["resp"]["id"] = event_id + + return event + + +class _V3IOMLRunEndPointClient(_MLRunEndPointClient): + """ + An MLRun model endpoint monitoring client to connect and send events on a V3IO stream. + """ + + def __init__( + self, + monitoring_stream_path: str, + monitoring_container: str, + model_endpoint_name: str, + model_endpoint_uid: str, + serving_function: str | RemoteRuntime, + serving_function_tag: str | None = None, + project: str | mlrun.projects.MlrunProject = None, + ): + """ + Initialize an MLRun model endpoint monitoring client. + + :param monitoring_stream_path: V3IO stream path. + :param monitoring_container: V3IO container name. + :param model_endpoint_name: The monitoring endpoint related model name. + :param model_endpoint_uid: Model endpoint unique identifier. + :param serving_function: Serving function name or ``RemoteRuntime`` object. + :param serving_function_tag: Optional function tag (defaults to 'latest'). + :param project: Project name or ``MlrunProject``. If ``None``, uses the current project. + raise: MLRunInvalidArgumentError: If there is no current active project and no `project` argument was provided. + """ + super().__init__( + model_endpoint_name=model_endpoint_name, + model_endpoint_uid=model_endpoint_uid, + serving_function=serving_function, + serving_function_tag=serving_function_tag, + project=project, + ) + + import v3io + + # Store the provided info: + self._monitoring_stream_path = monitoring_stream_path + self._monitoring_container = monitoring_container + + # Initialize a V3IO client: + self._v3io_client = v3io.Client() + + def monitor( + self, + event_id: str, + label: str, + input_data: dict, + output_data: dict, + request_timestamp: str, + response_timestamp: str, + ): + """ + Monitor the provided event, sending it to the model endpoint monitoring stream. + + :param event_id: Unique event identifier used as the monitored record id. + :param label: Label for the run/event. + :param input_data: Serialized input data for the run. + :param output_data: Serialized output data for the run. + :param request_timestamp: Request/start timestamp in the format of '%Y-%m-%d %H:%M:%S%z'. + :param response_timestamp: Response/end timestamp in the format of '%Y-%m-%d %H:%M:%S%z'. + """ + # Copy the sample: + event = self._create_event( + event_id=event_id, + label=label, + input_data=input_data, + output_data=output_data, + request_timestamp=request_timestamp, + response_timestamp=response_timestamp, + ) + + # Push to stream: + self._v3io_client.stream.put_records( + container=self._monitoring_container, + stream_path=self._monitoring_stream_path, + records=[{"data": orjson.dumps(event).decode('utf-8')}], + ) + + +class _KafkaMLRunEndPointClient(_MLRunEndPointClient): + """ + An MLRun model endpoint monitoring client to connect and send events on a Kafka stream. + """ + + def __init__( + self, + kafka_stream_profile_name: str, + model_endpoint_name: str, + model_endpoint_uid: str, + serving_function: str | RemoteRuntime, + serving_function_tag: str | None = None, + project: str | mlrun.projects.MlrunProject = None, + kafka_linger_ms: int = 0, + ): + """ + Initialize an MLRun model endpoint monitoring client for Kafka. + + :param kafka_stream_profile_name: The name of the registered DatastoreProfileKafkaStream to use for Kafka + configuration. This profile should be registered via ``project.register_datastore_profile()`` and + contains all Kafka settings including broker, topic, SASL credentials, SSL config, etc. + :param model_endpoint_name: The monitoring endpoint related model name. + :param model_endpoint_uid: Model endpoint unique identifier. + :param serving_function: Serving function name or ``RemoteRuntime`` object. + :param serving_function_tag: Optional function tag (defaults to 'latest'). + :param project: Project name or ``MlrunProject``. If ``None``, uses the current project. + :param kafka_linger_ms: Kafka producer linger.ms setting controlling message batching. Messages are + accumulated for up to this duration before being sent as a batch. Default: 500ms. + raise: MLRunInvalidArgumentError: If there is no current active project and no `project` argument was provided. + """ + super().__init__( + model_endpoint_name=model_endpoint_name, + model_endpoint_uid=model_endpoint_uid, + serving_function=serving_function, + serving_function_tag=serving_function_tag, + project=project, + ) + + from kafka import KafkaProducer + from mlrun.datastore.utils import KafkaParameters + from mlrun.common.model_monitoring.helpers import get_kafka_topic + + # Get project object using resolved project name from parent: + project_obj = mlrun.get_or_create_project(self._project_name) + + # Fetch the Kafka stream profile: + stream_profile = project_obj.get_datastore_profile(profile=kafka_stream_profile_name) + + # Get profile attributes and convert to producer config: + profile_attrs = stream_profile.attributes() + kafka_params = KafkaParameters(kwargs=profile_attrs) + producer_config = kafka_params.producer() + + # Extract broker and determine topic (use profile's topic if available, otherwise use MLRun's standard naming): + self._monitoring_broker = profile_attrs.get("brokers") + topics = profile_attrs.get("topics", []) + self._monitoring_topic = topics[0] if topics else get_kafka_topic(project=project_obj.name) + + # Remove bootstrap_servers from producer_config to avoid duplicate argument error: + producer_config.pop("bootstrap_servers", None) + + # Initialize a Kafka producer with full config from profile: + self._kafka_producer = KafkaProducer( + bootstrap_servers=self._monitoring_broker, + key_serializer=lambda k: k.encode("utf-8") if isinstance(k, str) else k, + value_serializer=( + lambda v: v if isinstance(v, bytes) + else orjson.dumps(v) if isinstance(v, dict) + else str(v).encode("utf-8") + ), + linger_ms=kafka_linger_ms, + **producer_config, + ) + + def monitor( + self, + event_id: str, + label: str, + input_data: dict, + output_data: dict, + request_timestamp: str, + response_timestamp: str, + ): + """ + Monitor the provided event, sending it to the model endpoint monitoring stream. + + :param event_id: Unique event identifier used as the monitored record id. + :param label: Label for the run/event. + :param input_data: Serialized input data for the run. + :param output_data: Serialized output data for the run. + :param request_timestamp: Request/start timestamp in the format of '%Y-%m-%d %H:%M:%S%z'. + :param response_timestamp: Response/end timestamp in the format of '%Y-%m-%d %H:%M:%S%z'. + """ + # Copy the sample: + event = self._create_event( + event_id=event_id, + label=label, + input_data=input_data, + output_data=output_data, + request_timestamp=request_timestamp, + response_timestamp=response_timestamp, + ) + + # Push to stream (async - message is buffered): + self._kafka_producer.send( + topic=self._monitoring_topic, + value=event, # Will be serialized by the value_serializer + key=self._model_endpoint_uid, + ) + + def flush(self): + """ + Flush all buffered messages to ensure they are sent to Kafka. + + Blocks until all buffered messages are delivered and acknowledged by the broker. + """ + self._kafka_producer.flush() + + +class MLRunTracerClientSettings(BaseSettings): + """ + MLRun tracer monitoring client configurations. These are mandatory arguments for allowing MLRun to send monitoring + events to a specific model endpoint stream. + """ + + v3io_stream_path: str | None = None + """ + The V3IO stream path to send the events to. + """ + + v3io_container: str | None = None + """ + The V3IO stream container. + """ + + kafka_stream_profile_name: str | None = None + """ + The name of the registered DatastoreProfileKafkaStream to use for Kafka configuration. This profile should be + registered via ``project.register_datastore_profile()`` and contains all Kafka settings including broker, topic, + SASL credentials, SSL config, etc. + """ + + kafka_linger_ms: int = 500 + """ + The Kafka producer linger.ms setting controlling message batching (in milliseconds). Messages are accumulated for + up to this duration before being sent as a batch, reducing network overhead. + + The tracer always flushes at the end of each root run, guaranteeing delivery regardless of this setting. + Default: 500ms. Set to 0 to disable batching (each message sent immediately). + """ + + model_endpoint_name: str = ... + """ + The model endpoint name. + """ + + model_endpoint_uid: str = ... + """ + The model endpoint UID. + """ + + serving_function: str = ... + """ + The serving function name. + """ + + serving_function_tag: str | None = None + """ + The serving function tag. If not set, it will be 'latest' by default. + """ + + project: str | None = None + """ + The MLRun project name related to the serving function and model endpoint. + """ + + #: Pydantic model configuration to set the environment variable prefix. + model_config = SettingsConfigDict(env_prefix="LC_MLRUN_TRACER_CLIENT_") + + @model_validator(mode='after') + def validate_stream_settings(self) -> 'MLRunTracerClientSettings': + """ + Validate that either V3IO settings or stream profile name is provided, but not both or none. + + :returns: The validated settings instance. + """ + v3io_settings = all([self.v3io_container, self.v3io_stream_path]) + kafka_settings = self.kafka_stream_profile_name is not None + + if v3io_settings and kafka_settings: + raise ValueError("Provide either V3IO settings OR Kafka settings, not both.") + if not v3io_settings and not kafka_settings: + raise ValueError("You must provide either a complete V3IO settings or complete Kafka settings. See docs for more information") + return self + +class MLRunTracerMonitorSettings(BaseSettings): + """ + MLRun tracer monitoring configurations. These are optional arguments to customize the LangChain runs summarization + into monitorable MLRun endpoint events. If needed, a custom summarization can be passed. + """ + + label: str = "default" + """ + Label to use for all monitored runs. Can be used to differentiate between different monitored sources on the same + endpoint. + """ + + tags_filter: list[str] | None = None + """ + Filter runs by tags. Only runs with at least one tag in this list will be monitored. + If None, no tag-based filtering is applied and runs with any tags are considered. + Default: None. + """ + + run_types_filter: list[str] | None = None + """ + Filter runs by run types (e.g. "chain", "llm", "chat", "tool"). + Only runs whose `run_type` appears in this list will be monitored. + If None, no run-type filtering is applied. + Default: None. + """ + + names_filter: list[str] | None = None + """ + Filter runs by class/name. Only runs whose `name` appears in this list will be monitored. + If None, no name-based filtering is applied. + Default: None. + """ + + include_full_run: bool = False + """ + If True, include the complete serialized run dict (the output of `run._get_dicts_safe()`) + in the event outputs under the key `full_run`. Useful for debugging or when consumers need + the raw run payload. Default: False. + """ + + include_errors: bool = True + """ + If True, include run error information in the outputs under the `error` key. + If False, runs that contain an error may be skipped by the summarizer filters. + Default: True. + """ + + include_metadata: bool = True + """ + If True, include run metadata (environment, tool metadata, etc.) in the inputs under + the `metadata` key. Default: True. + """ + + include_latency: bool = True + """ + If True, include latency information in the outputs under the `latency` key. + Default: True. + """ + + root_run_only: bool = False + """ + If True, only the root/top-level run will be monitored and any child runs will be + ignored/removed from monitoring. Use when only the top-level run should produce events. + Default: False. + """ + + split_runs: bool = False + """ + If True, child runs are emitted as separate monitoring events (each run summarized and + sent individually). If False, child runs are nested inside the parent/root run event under + `child_runs`. Default: False. + """ + + run_summarizer_function: ( + str + | Callable[ + [Run, Optional[BaseSettings]], + Generator[tuple[dict, dict] | None, None, None], + ] + | None + ) = None + """ + A function to summarize a `Run` object into a tuple of inputs and outputs. Can be passed directly or via a full + module path ("a.b.c.my_summarizer" will be imported as `from a.b.c import my_summarizer`). + + A summarizer is a function that will be used to process a run into monitoring events. The function is expected to be + of type: + `Callable[[Run, Optional[BaseSettings]], Generator[tuple[dict, dict] | None, None, None]]`, meaning + get a run object and optionally a settings object and return a generator yielding tuples of serialized dictionaries, + the (inputs, outputs) to send to MLRun monitoring as events or `None` to skip monitoring this run. + """ + + run_summarizer_settings: str | BaseSettings | None = None + """ + Settings to pass to the run summarizer function. Can be passed directly or via a full module path to be imported + and initialized. If the summarizer function does not require settings, this can be left as None. + """ + + debug: bool = False + """ + If True, disable sending events to MLRun and instead route events to `debug_target_list` + or print them as JSON to stdout. Useful for unit tests and local debugging. Default: False. + """ + + debug_target_list: list[dict] | bool = False + """ + Optional list to which debug events will be appended when `debug` is True. + If set, each generated event dict will be appended to this list. If not set and `debug` is True, + events will be printed to stdout as JSON. Default: False. + """ + + #: Pydantic model configuration to set the environment variable prefix. + model_config = SettingsConfigDict(env_prefix="LC_MLRUN_TRACER_MONITOR_") + + @field_validator('debug_target_list', mode='before') + @classmethod + def convert_bool_to_list(cls, v): + """ + Convert a boolean `True` value to an empty list for `debug_target_list`. + + :param v: The value to validate. + + :returns: An empty list if `v` is True, otherwise the original value. + """ + if v is True: + return [] + return v + + +class MLRunTracerSettings(BaseSettings): + """ + MLRun tracer settings to configure the tracer. The settings are split into two groups: + + * `client`: settings required to connect and send events to the MLRun monitoring stream. + * `monitor`: settings controlling which LangChain runs are summarized and sent and how. + """ + + client: MLRunTracerClientSettings = Field(default_factory=MLRunTracerClientSettings) + """ + Client configuration group (``MLRunTracerClientSettings``). + + Contains the mandatory connection and endpoint information required to publish monitoring + events. Values may be supplied programmatically or via environment variables prefixed with + `LC_MLRUN_TRACER_CLIENT_`. See more at ``MLRunTracerClientSettings``. + """ + + monitor: MLRunTracerMonitorSettings = Field(default_factory=MLRunTracerMonitorSettings) + """ + Monitoring configuration group (``MLRunTracerMonitorSettings``). + + Controls what runs are captured, how they are summarized (including custom summarizer import + options), whether child runs are split or nested, and debug behavior. Values may be supplied + programmatically or via environment variables prefixed with `LC_MLRUN_TRACER_MONITOR_`. + See more at ``MLRunTracerMonitorSettings``. + """ + + #: Pydantic model configuration to set the environment variable prefix. + model_config = SettingsConfigDict(env_prefix="LC_MLRUN_TRACER_") + + +class MLRunTracer(BaseTracer): + """ + MLRun tracer for LangChain runs allowing monitoring LangChain and LangGraph in production using MLRun's monitoring. + + There are two usage modes for the MLRun tracer following LangChain tracing best practices: + + 1. **Manual Mode** - Using the ``mlrun_monitoring`` context manager:: + + from mlrun_tracer import mlrun_monitoring + + with mlrun_monitoring(...) as tracer: + # LangChain code here. + pass + + 2. **Auto Mode** - Setting the `LC_MLRUN_MONITORING_ENABLED="1"` environment variable:: + + import mlrun_integration.tracer + + # All LangChain code will be automatically traced and monitored. + pass + + To control how runs are being summarized into the events being monitored, the ``MLRunTracerSettings`` can be set. + As it is a Pydantic ``BaseSettings`` class, it can be done in two ways: + + 1. Initializing the settings classes and passing them to the context manager:: + + from mlrun_tracer import ( + mlrun_monitoring, + MLRunTracerSettings, + MLRunTracerClientSettings, + MLRunTracerMonitorSettings, + ) + + my_settings = MLRunTracerSettings( + client=MLRunTracerClientSettings(), + monitor=MLRunTracerMonitorSettings(root_run_only=True), + ) + + with mlrun_monitoring(settings=my_settings) as tracer: + # LangChain code here. + pass + + 2. Or via environment variables following the prefix 'LC_MLRUN_TRACER_CLIENT_' for client settings and + 'LC_MLRUN_TRACER_MONITOR_' for monitoring settings. + """ + + #: A singleton tracer for when using the tracer via environment variable to activate global tracing. + _singleton_tracer: "MLRunTracer | None" = None + #: A thread lock for initializing the tracer singleton safely. + _lock = threading.Lock() + #: A boolean flag to know whether the singleton was initialized. + _initialized = False + + def __new__(cls, *args, **kwargs) -> "MLRunTracer": + """ + Create or return an ``MLRunTracer`` instance. + + When ``LC_MLRUN_MONITORING_ENABLED`` is not set to ``"1"``, a normal instance is returned. + When the env var is ``"1"``, a process-wide singleton is returned. Creation is thread-safe. + + :returns: MLRunTracer instance (singleton if 'auto' mode is active). + """ + # Check if needed to use a singleton as the user is using the MLRun tracer by setting the environment variable + # and not manually (via context manager): + if not cls._check_for_env_var_usage(): + return super(MLRunTracer, cls).__new__(cls) + + # Check if the singleton is set: + if cls._singleton_tracer is None: + # Acquire lock to initialize the singleton: + with cls._lock: + # Double-check after acquiring lock: + if cls._singleton_tracer is None: + cls._singleton_tracer = super(MLRunTracer, cls).__new__(cls) + + return cls._singleton_tracer + + def __init__(self, settings: MLRunTracerSettings = None, **kwargs): + """ + Initialize the tracer. + + :param settings: Settings to use for the tracer. If not passed, defaults are used and environment variables are + applied per Pydantic settings behavior. + :param kwargs: Passed to the base initializer. + """ + # Proceed with initialization only if singleton mode is not required or the singleton was not initialized: + if self._check_for_env_var_usage() and self._initialized: + return + + # Call the base tracer init: + super().__init__(**kwargs) + + # Set a UID for this instance: + self._uid = uuid7() + + # Set the settings: + self._settings = settings or MLRunTracerSettings() + self._client_settings = self._settings.client + self._monitor_settings = self._settings.monitor + + # Initialize the MLRun endpoint client: + self._mlrun_client = ( + self._get_mlrun_client() + if not self._monitor_settings.debug + else None + ) + + # In case the user passed a custom summarizer, import it: + self._custom_run_summarizer_function: ( + Callable[ + [Run, Optional[BaseSettings]], + Generator[tuple[dict, dict] | None, None, None], + ] + | None + ) = None + self._custom_run_summarizer_settings: BaseSettings | None = None + self._import_custom_run_summarizer() + + # Mark the initialization flag (for the singleton case): + self._initialized = True + + @property + def settings(self) -> MLRunTracerSettings: + """ + Access the effective settings. + + :returns: The settings used by this tracer. + """ + return self._settings + + def _get_mlrun_client(self) -> _MLRunEndPointClient: + """ + Create and return an MLRun model endpoint monitoring client based on the MLRun (CE or not) and current + configuration. + + :returns: An MLRun model endpoint monitoring client. + """ + if mlrun.mlconf.is_ce_mode(): + return _KafkaMLRunEndPointClient( + kafka_stream_profile_name=self._client_settings.kafka_stream_profile_name, + model_endpoint_name=self._client_settings.model_endpoint_name, + model_endpoint_uid=self._client_settings.model_endpoint_uid, + serving_function=self._client_settings.serving_function, + serving_function_tag=self._client_settings.serving_function_tag, + project=self._client_settings.project, + kafka_linger_ms=self._client_settings.kafka_linger_ms, + ) + return _V3IOMLRunEndPointClient( + monitoring_stream_path=self._client_settings.v3io_stream_path, + monitoring_container=self._client_settings.v3io_container, + model_endpoint_name=self._client_settings.model_endpoint_name, + model_endpoint_uid=self._client_settings.model_endpoint_uid, + serving_function=self._client_settings.serving_function, + serving_function_tag=self._client_settings.serving_function_tag, + project=self._client_settings.project, + ) + + def _import_custom_run_summarizer(self): + """ + Import or assign a custom run summarizer (and its custom settings) if configured. + """ + # If the user did not pass a run summarizer function, return: + if not self._monitor_settings.run_summarizer_function: + return + + # Check if the function needs to be imported: + if isinstance(self._monitor_settings.run_summarizer_function, str): + self._custom_run_summarizer_function = self._import_from_module_path( + module_path=self._monitor_settings.run_summarizer_function + ) + else: + self._custom_run_summarizer_function = ( + self._monitor_settings.run_summarizer_function + ) + + # Check if the user passed settings as well: + if self._monitor_settings.run_summarizer_settings: + # Check if the settings need to be imported: + if isinstance(self._monitor_settings.run_summarizer_settings, str): + self._custom_run_summarizer_settings = self._import_from_module_path( + module_path=self._monitor_settings.run_summarizer_settings + )() + else: + self._custom_run_summarizer_settings = ( + self._monitor_settings.run_summarizer_settings + ) + + def _persist_run(self, run: Run, level: int = 0) -> None: + """ + Summarize the run (and its children) into MLRun monitoring events. + + Note: This will use the MLRun tracer's default summarization that can be configured via + ``MLRunTracerMonitorSettings``, unless a custom summarizer was provided (via the same settings). + + :param run: LangChain run object to process holding all the nested tree of runs. + :param level: The nesting level of the run (0 for root runs, incremented for child runs). + """ + try: + # Serialize the run: + serialized_run = self._serialize_run( + run=run, + include_child_runs=not (self._settings.monitor.root_run_only or self._settings.monitor.split_runs) + ) + + # Check for a user custom run summarizer function: + if self._custom_run_summarizer_function: + for summarized_run in self._custom_run_summarizer_function( + run, self._custom_run_summarizer_settings + ): + if summarized_run: + inputs, outputs = summarized_run + self._send_run_event( + event_id=serialized_run["id"], + inputs=inputs, + outputs=outputs, + start_time=run.start_time, + end_time=run.end_time, + ) + return + + # Check how to deal with the child runs, monitor them in separate events or as a single event: + if self._monitor_settings.split_runs and not self._settings.monitor.root_run_only: + # Monitor as separate events: + for child_run in run.child_runs: + self._persist_run(run=child_run, level=level + 1) + summarized_run = self._summarize_run(serialized_run=serialized_run, include_children=False) + if summarized_run: + inputs, outputs = summarized_run + inputs["child_level"] = level + self._send_run_event( + event_id=serialized_run["id"], + inputs=inputs, + outputs=outputs, + start_time=run.start_time, + end_time=run.end_time, + ) + return + + # Monitor the root event (include child runs if `root_run_only` is False): + summarized_run = self._summarize_run( + serialized_run=serialized_run, + include_children=not self._monitor_settings.root_run_only + ) + if not summarized_run: + return + inputs, outputs = summarized_run + inputs["child_level"] = level + self._send_run_event( + event_id=serialized_run["id"], + inputs=inputs, + outputs=outputs, + start_time=run.start_time, + end_time=run.end_time, + ) + finally: + # Flush buffered messages after root run completion to ensure delivery: + if level == 0 and self._mlrun_client: + self._mlrun_client.flush() + + def _serialize_run(self, run: Run, include_child_runs: bool) -> dict: + """ + Serialize a LangChain run into a dictionary. + + :param run: The run to serialize. + :param include_child_runs: Whether to include child runs in the serialization. + + :returns: The serialized run dictionary. + """ + # In LangChain 1.2.3+, the Run model uses Pydantic v2 with child_runs marked as Field(exclude=True), so we + # must manually serialize child runs. Still excluding manually for future compatibility. In previous + # LangChain versions, Run was Pydantic v1, so we use dict. + serialized_run = ( + run.model_dump(exclude={"child_runs"}) + if hasattr(run, "model_dump") + else run.dict(exclude={"child_runs"}) + ) + + # Manually serialize child runs if needed: + if include_child_runs and run.child_runs: + serialized_run["child_runs"] = [ + self._serialize_run(child_run, include_child_runs=True) + for child_run in run.child_runs + ] + + return orjson.loads(orjson.dumps(serialized_run, default=self._serialize_default)) + + def _serialize_default(self, obj: Any): + """ + Default serializer for objects present in LangChain run that are not serializable by default JSON encoder. It + includes handling Pydantic v1 and v2 models, UUIDs, and datetimes. + + :param obj: The object to serialize. + + :returns: The serialized object. + """ + if isinstance(obj, UUID): + return str(obj) + if isinstance(obj, datetime.datetime): + return obj.isoformat() + if hasattr(obj, "model_dump"): + return orjson.loads(orjson.dumps(obj.model_dump(), default=self._serialize_default)) + if hasattr(obj, "dict"): + return orjson.loads(orjson.dumps(obj.dict(), default=self._serialize_default)) + return str(obj) + + def _filter_by_tags(self, serialized_run: dict) -> bool: + """ + Apply tag-based filtering. + + :param serialized_run: Serialized run dictionary. + + :returns: True if the run passes tag filters or if no tag filter is configured. + """ + # Check if the user enabled filtering by tags: + if not self._monitor_settings.tags_filter: + return True + + # Filter the run: + return not set(self._monitor_settings.tags_filter).isdisjoint( + serialized_run["tags"] + ) + + def _filter_by_run_types(self, serialized_run: dict) -> bool: + """ + Apply run-type filtering. + + :param serialized_run: Serialized run dictionary. + + :returns: True if the run's ``run_type`` is allowed or if no run-type filter is configured. + """ + # Check if the user enabled filtering by run types: + if not self._monitor_settings.run_types_filter: + return True + + # Filter the run: + return serialized_run["run_type"] in self._monitor_settings.run_types_filter + + def _filter_by_names(self, serialized_run: dict) -> bool: + """ + Apply class/name filtering. + + :param serialized_run: Serialized run dictionary. + + :returns: True if the run's ``name`` is allowed or if no name filter is configured. + """ + # Check if the user enabled filtering by class names: + if not self._monitor_settings.names_filter: + return True + + # Filter the run: + return serialized_run["name"] in self._monitor_settings.names_filter + + def _get_run_inputs(self, serialized_run: dict) -> dict[str, Any]: + """ + Build the inputs dictionary for a monitoring event. + + :param serialized_run: Serialized run dictionary. + + :returns: A dictionary containing inputs, run metadata and (optionally) additional metadata. + """ + inputs = { + "inputs": serialized_run["inputs"], + "run_type": serialized_run["run_type"], + "run_name": serialized_run["name"], + "tags": serialized_run["tags"], + "run_id": serialized_run["id"], + "start_timestamp": serialized_run["start_time"], + } + if "parent_run_id" in serialized_run: + # Parent run ID is excluded when child runs are joined in the same event. When child runs are split, it is + # included and can be used to reconstruct the run tree if needed. + inputs = {**inputs, "parent_run_id": serialized_run["parent_run_id"]} + if self._monitor_settings.include_metadata and "metadata" in serialized_run: + inputs = {**inputs, "metadata": serialized_run["metadata"]} + + return inputs + + def _get_run_outputs(self, serialized_run: dict) -> dict[str, Any]: + """ + Build the outputs dictionary for a monitoring event. + + :param serialized_run: Serialized run dictionary. + + :returns: A dictionary with outputs and optional other collected info depending on monitor settings. + """ + outputs = {"outputs": serialized_run["outputs"], "end_timestamp": serialized_run["end_time"]} + if self._monitor_settings.include_latency and "latency" in serialized_run: + outputs = {**outputs, "latency": serialized_run["latency"]} + if self._monitor_settings.include_errors: + outputs = {**outputs, "error": serialized_run["error"]} + if self._monitor_settings.include_full_run: + outputs = {**outputs, "full_run": serialized_run} + + return outputs + + def _summarize_run(self, serialized_run: dict, include_children: bool) -> tuple[dict, dict] | None: + """ + Summarize a single run into (inputs, outputs) if it passes filters. + + :param serialized_run: Serialized run dictionary. + :param include_children: Whether to include child runs. + + :returns: The summarized run (inputs, outputs) tuple if the run should be monitored, otherwise ``None``. + """ + # Pass filters: + if not ( + self._filter_by_tags(serialized_run=serialized_run) + and self._filter_by_run_types(serialized_run=serialized_run) + and self._filter_by_names(serialized_run=serialized_run) + ): + return None + + # Check if needed to include errors: + if serialized_run["error"] and not self._monitor_settings.include_errors: + return None + + # Prepare the inputs and outputs: + inputs = self._get_run_inputs(serialized_run=serialized_run) + outputs = self._get_run_outputs(serialized_run=serialized_run) + + # Check if needed to include child runs: + if include_children: + outputs["child_runs"] = [] + for child_run in serialized_run.get("child_runs", []): + # Recursively summarize the child run: + summarized_child_run = self._summarize_run(serialized_run=child_run, include_children=True) + if summarized_child_run: + inputs_child, outputs_child = summarized_child_run + outputs["child_runs"].append( + { + "input_data": inputs_child, + "output_data": outputs_child, + } + ) + + return inputs, outputs + + def _send_run_event( + self, event_id: str, inputs: dict, outputs: dict, start_time: datetime.datetime, end_time: datetime.datetime + ): + """ + Send a monitoring event for a single run. + + Note: If monitor debug mode is enabled, appends to ``debug_target_list`` or prints JSON. + + :param event_id: Unique event identifier. + :param inputs: Inputs dictionary for the event. + :param outputs: Outputs dictionary for the event. + :param start_time: Request/start timestamp. + :param end_time: Response/end timestamp. + """ + event = { + "event_id": event_id, + "label": self._monitor_settings.label, + "input_data": {"input_data": inputs}, # So it will be a single "input feature" in MLRun monitoring. + "output_data": {"output_data": outputs}, # So it will be a single "output feature" in MLRun monitoring. + "request_timestamp": start_time.strftime("%Y-%m-%d %H:%M:%S%z"), + "response_timestamp": end_time.strftime("%Y-%m-%d %H:%M:%S%z"), + } + if self._monitor_settings.debug: + if isinstance(self._monitor_settings.debug_target_list, list): + self._monitor_settings.debug_target_list.append(event) + else: + print(orjson.dumps(event, option=orjson.OPT_INDENT_2 | orjson.OPT_APPEND_NEWLINE)) + return + + self._mlrun_client.monitor(**event) + + @staticmethod + def _check_for_env_var_usage() -> bool: + """ + Check whether global env-var activated tracing is requested. + + :returns: True when ``LC_MLRUN_MONITORING_ENABLED`` environment variable equals ``"1"``. + """ + return os.environ.get(mlrun_monitoring_env_var, "0") == "1" + + @staticmethod + def _import_from_module_path(module_path: str) -> Any: + """ + Import an object from a full module path string. + + :param module_path: Full dotted path, e.g. ``a.b.module.object``. + + :returns: The imported object. + """ + try: + module_name, object_name = module_path.rsplit(".", 1) + module = importlib.import_module(module_name) + obj = getattr(module, object_name) + except ValueError as value_error: + raise ValueError( + f"The provided '{module_path}' is not valid: it must have at least one '.'. " + f"If the class is locally defined, please add '__main__.MyObject' to the path." + ) from value_error + except ImportError as import_error: + raise ImportError( + f"Could not import '{module_path}'. Tried to import '{module_name}' and failed with the following " + f"error: {import_error}." + ) from import_error + except AttributeError as attribute_error: + raise AttributeError( + f"Could not import '{object_name}'. Tried to run 'from {module_name} import {object_name}' and could " + f"not find it: {attribute_error}" + ) from attribute_error + + return obj + + +#: MLRun monitoring context variable to set when the user wraps his code with `mlrun_monitoring`. From this context +# variable LangChain will get the tracer in a thread-safe way. +mlrun_monitoring_var: ContextVar[MLRunTracer | None] = ContextVar( + "mlrun_monitoring", default=None +) + + +@contextmanager +def mlrun_monitoring(settings: MLRunTracerSettings | None = None): + """ + Context manager to enable MLRun tracing for LangChain code to monitor LangChain runs. + + Example usage:: + + from mlrun_tracer import mlrun_monitoring, MLRunTracerSettings + + settings = MLRunTracerSettings(...) + with mlrun_monitoring(settings=settings) as tracer: + # LangChain execution within this block will be traced by `tracer`. + ... + + :param settings: The settings to use to configure the tracer. + """ + mlrun_tracer = MLRunTracer(settings=settings) + token = mlrun_monitoring_var.set(mlrun_tracer) + try: + yield mlrun_tracer + finally: + mlrun_monitoring_var.reset(token) + + +# Register a hook for LangChain to apply the MLRun tracer: +register_configure_hook( + context_var=mlrun_monitoring_var, + inheritable=True, # To allow inner runs (agent that uses a tool that uses a llm...) to be traced. + env_var=mlrun_monitoring_env_var, + handle_class=MLRunTracer, +) + +# Temporary convenient function to set up the monitoring infrastructure required for the tracer. +def setup_langchain_monitoring( + project: str | mlrun.MlrunProject = None, + function_name: str = "langchain_mlrun_function", + model_name: str = "langchain_mlrun_model", + model_endpoint_name: str = "langchain_mlrun_endpoint", + v3io_container: str = "projects", + v3io_stream_path: str = None, + kafka_stream_profile_name: str = None, + kafka_linger_ms: int = 500, +) -> dict: + """ + Create a model endpoint in the given project to be used for LangChain monitoring with MLRun and returns the + necessary environment variables to configure the MLRun tracer client. The project should already exist and have + monitoring enabled:: + + project.set_model_monitoring_credentials( + stream_profile_name=..., + tsdb_profile_name=... + ) + + This function creates and logs dummy model and function in the specified project in order to create the model + endpoint for monitoring. It is a temporary workaround and will be added as a feature in a future MLRun version. + + :param project: The MLRun project name or object where to create the model endpoint. If None, the current active + project will be used. + :param function_name: The name of the serving function to create. + :param model_name: The name of the model to create. + :param model_endpoint_name: The name of the model endpoint to create. + :param v3io_container: The V3IO container where the monitoring stream is located (for MLRun Enterprise). + :param v3io_stream_path: The V3IO stream path for monitoring (for MLRun Enterprise). If None, + ``/model-endpoints/stream-v1`` will be used. + :param kafka_stream_profile_name: The name of the registered ``DatastoreProfileKafkaStream`` to use for Kafka + configuration (required for MLRun CE). This profile should be registered via + ``project.register_datastore_profile()`` and contains all Kafka settings including broker, topic, + SASL credentials, SSL config, etc. + :param kafka_linger_ms: Kafka producer linger.ms setting controlling message batching (default: 500ms). + Messages are accumulated for up to this duration before being sent as a batch, reducing network overhead. + The tracer always flushes at the end of each root run, guaranteeing delivery. Set to 0 to disable batching. + + :returns: A dictionary with the necessary environment variables to configure the MLRun tracer client. + raise: MLRunInvalidArgumentError: If no project is provided and there is no current active project. + """ + import io + import time + import sys + from contextlib import redirect_stdout, redirect_stderr + import tempfile + import pickle + import json + + from mlrun.features import Feature + + class ProgressStep: + """ + A context manager to display progress of a code block with timing and optional output suppression. + """ + + def __init__(self, label: str, indent: int = 2, width: int = 40, clean: bool = True): + """ + Initialize the ProgressStep context manager. + + :param label: The label to display for the progress step. + :param indent: The number of spaces to indent the label. + :param width: The width to pad the label for alignment. + :param clean: Whether to suppress stdout and stderr during the block execution. + """ + # Store parameters: + self._label = label + self._indent = indent + self._width = width + self._clean = clean + + # Internal state: + self._start_time = None + self._sink = io.StringIO() + self._stdout_redirect = None + self._stderr_redirect = None + self._last_line_length = 0 # To track the line printed when terminals don't support '\033[K'. + + # Capture the stream currently in use (before and if clean is true and we redirect it): + self._terminal = sys.stdout + + def __enter__(self): + """ + Enter the context manager, starting the timer and printing the initial status. + """ + # Start timer: + self._start_time = time.perf_counter() + + # Print without newline (using \r to allow overwriting): + self._write(icon=" ", status="Running", new_line=False) + + # Silence all internal noise: + if self._clean: + self._stdout_redirect = redirect_stdout(self._sink) + self._stderr_redirect = redirect_stderr(self._sink) + self._stdout_redirect.__enter__() + self._stderr_redirect.__enter__() + + return self + + def __exit__(self, exc_type, exc_val, exc_tb): + """ + Exit the context manager, stopping the timer and printing the final status. + + :param exc_type: The exception type, if any. + :param exc_val: The exception value, if any. + :param exc_tb: The exception traceback, if any. + """ + # Restore stdout/stderr: + if self._clean: + self._stdout_redirect.__exit__(exc_type, exc_val, exc_tb) + self._stderr_redirect.__exit__(exc_type, exc_val, exc_tb) + + # Calculate elapsed time: + elapsed = time.perf_counter() - self._start_time + + # Move cursor back to start of line ('\r') and overwrite ('\033[K' clears the line to the right): + if exc_type is None: + self._write(icon="✓", status=f"Done ({elapsed:.2f}s)", new_line=True) + else: + self._write(icon="✕", status="Failed", new_line=True) + + def update(self, status: str): + """ + Update the status message displayed for the progress step. + + :param status: The new status message to display. + """ + self._write(icon=" ", status=status, new_line=False) + + def _write(self, icon: str, status: str, new_line: bool): + """ + Write the progress line to the terminal, handling line clearing for terminals that do not support it. + + :param icon: The icon to display (e.g., checkmark, cross, space). + :param status: The status message to display. + :param new_line: Whether to end the line with a newline character. + """ + # Construct the basic line + line = f"\r{' ' * self._indent}[{icon}] {self._label.ljust(self._width, '.')} {status}" + + # Calculate if we need to pad with spaces to clear the old, longer line: + padding = max(0, self._last_line_length - len(line)) + + # Add spaces to clear old text (add the ANSI clear for terminals that support it): + line = f"{line}{' ' * padding}\033[K" + + # Add newline if needed: + if new_line: + line += "\n" + + # Write to terminal: + self._terminal.write(line) + self._terminal.flush() + + # Update the max length seen so far: + self._last_line_length = len(line) + + print("Creating LangChain model endpoint\n") + + # Get the project: + with ProgressStep("Loading Project"): + if project is None: + try: + project = mlrun.get_current_project(silent=False) + except mlrun.errors.MLRunInvalidArgumentError: + raise mlrun.errors.MLRunInvalidArgumentError( + "There is no current active project. Either use `mlrun.get_or_create_project` prior to " + "creating the monitoring endpoint or pass a project name to load." + ) + if isinstance(project, str): + project = mlrun.load_project(name=project) + + # Create and log the dummy model: + with ProgressStep(f"Creating Model") as progress_step: + # Check if the model already exists: + progress_step.update("Checking if model exists") + try: + dummy_model = project.get_artifact(key=model_name) + except mlrun.MLRunNotFoundError: + dummy_model = None + # If not, create and log it: + if not dummy_model: + progress_step.update(f"Logging model '{model_name}'") + with tempfile.TemporaryDirectory() as tmpdir: + # Create a dummy model file: + dummy_model_path = os.path.join(tmpdir, "for_langchain_mlrun_tracer.pkl") + with open(dummy_model_path, "wb") as f: + pickle.dump({"dummy": "model"}, f) + # Log the model: + dummy_model = project.log_model( + key=model_name, + model_file=dummy_model_path, + inputs=[Feature(value_type="str", name="input")], + outputs=[Feature(value_type='str', name="output")] + ) + + # Create and set the dummy function: + with ProgressStep("Creating Function") as progress_step: + # Check if the function already exists: + progress_step.update("Checking if function exists") + try: + dummy_function = project.get_function(key=function_name) + except mlrun.MLRunNotFoundError: + dummy_function = None + # If not, create and save it: + if not dummy_function: + progress_step.update(f"Setting function '{function_name}'") + with tempfile.TemporaryDirectory() as tmpdir: + # Create a dummy function file: + dummy_function_code = """ +def handler(context, event): + return "ok" +""" + dummy_function_path = os.path.join(tmpdir, "dummy_function.py") + with open(dummy_function_path, "w") as f: + f.write(dummy_function_code) + # Set the function in the project: + dummy_function = project.set_function( + func=dummy_function_path, name=function_name, image="mlrun/mlrun", kind="nuclio" + ) + dummy_function.save() + + # Create the model endpoint: + with ProgressStep("Creating Model Endpoint") as progress_step: + # Get the MLRun DB: + progress_step.update("Getting MLRun DB") + db = mlrun.get_run_db() + # Check if the model endpoint already exists: + progress_step.update("Checking if endpoint exists") + model_endpoint = project.list_model_endpoints(names=[model_endpoint_name]).endpoints + if model_endpoint: + model_endpoint = model_endpoint[0] + else: + progress_step.update("Creating model endpoint") + model_endpoint = mlrun.common.schemas.ModelEndpoint( + metadata=mlrun.common.schemas.ModelEndpointMetadata( + project=project.name, + name=model_endpoint_name, + endpoint_type=mlrun.common.schemas.model_monitoring.EndpointType.NODE_EP, + ), + spec=mlrun.common.schemas.ModelEndpointSpec( + function_name=dummy_function.metadata.name, + function_tag="latest", + model_path=dummy_model.uri, + model_class="CustomStream", + ), + status=mlrun.common.schemas.ModelEndpointStatus( + monitoring_mode=mm_constants.ModelMonitoringMode.enabled, + ), + ) + db.create_model_endpoint(model_endpoint=model_endpoint) + # Wait for the model endpoint UID to be set: + progress_step.update("Waiting for model endpoint") + uid_exist_flag = False + while not uid_exist_flag: + model_endpoint = project.list_model_endpoints(names=[model_endpoint_name]) + model_endpoint = model_endpoint.endpoints[0] + if model_endpoint.metadata.uid: + uid_exist_flag = True + + # Set parameters defaults: + v3io_stream_path = v3io_stream_path or f"{project.name}/model-endpoints/stream-v1" + + if mlrun.mlconf.is_ce_mode(): + if kafka_stream_profile_name is None: + raise ValueError( + "kafka_stream_profile_name is required for MLRun CE mode. Register a DatastoreProfileKafkaStream and " + "pass its name." + ) + client_env_vars = { + "LC_MLRUN_TRACER_CLIENT_KAFKA_STREAM_PROFILE_NAME": kafka_stream_profile_name, + "LC_MLRUN_TRACER_CLIENT_KAFKA_LINGER_MS": str(kafka_linger_ms), + } + else: + client_env_vars = { + "LC_MLRUN_TRACER_CLIENT_V3IO_STREAM_PATH": v3io_stream_path, + "LC_MLRUN_TRACER_CLIENT_V3IO_CONTAINER": v3io_container, + } + + # Prepare the environment variables: + env_vars = { + "LC_MLRUN_MONITORING_ENABLED": "1", + "LC_MLRUN_TRACER_CLIENT_PROJECT": project.name, + "LC_MLRUN_TRACER_CLIENT_MODEL_ENDPOINT_NAME": model_endpoint.metadata.name, + "LC_MLRUN_TRACER_CLIENT_MODEL_ENDPOINT_UID": model_endpoint.metadata.uid, + "LC_MLRUN_TRACER_CLIENT_SERVING_FUNCTION": function_name, + **client_env_vars + } + print("\n✨ Done! LangChain monitoring model endpoint created successfully.") + print("You can now set the following environment variables to enable MLRun tracing in your LangChain code:\n") + print(json.dumps(env_vars, indent=4)) + print( + "\nTo customize the monitoring behavior, you can also set additional environment variables prefixed with " + "'LC_MLRUN_TRACER_MONITOR_'. Refer to the MLRun tracer documentation for more details.\n" + ) + + return env_vars + + +class LangChainMonitoringApp(ModelMonitoringApplicationBase): + """ + A base monitoring application for LangChain that calculates common metrics on LangChain runs traced with the MLRun + tracer. + + The class is inheritable and can be extended to add custom metrics or override existing ones. It provides methods to + extract structured runs from the monitoring context and calculate metrics such as average latency, success rate, + token usage, and run name counts. + + If inheriting, the main method to override is `do_tracking`, which performs the tracking on the monitoring context. + """ + + def do_tracking(self, monitoring_context: MonitoringApplicationContext) -> ( + ModelMonitoringApplicationResult | + list[ModelMonitoringApplicationResult | ModelMonitoringApplicationMetric] | + dict[str, Any] + ): + """ + The main function that performs tracking on the monitoring context. The LangChain monitoring app by default + will calculate all the provided metrics on the structured runs extracted from the monitoring context sample + dataframe. + + :param monitoring_context: The monitoring context containing the sample dataframe. + + :returns: The monitoring artifacts, metrics and results. + """ + # Get the structured runs from the monitoring context: + structured_runs, _ = self.get_structured_runs(monitoring_context=monitoring_context) + + # Calculate the metrics: + average_latency = self.calculate_average_latency(structured_runs=structured_runs) + success_rate = self.calculate_success_rate(structured_runs=structured_runs) + token_usage = self.count_token_usage(structured_runs=structured_runs) + run_name_counts = self.count_run_names(structured_runs=structured_runs) + + return [ + ModelMonitoringApplicationMetric( + name="average_latency", + value=average_latency, + ), + ModelMonitoringApplicationMetric( + name="success_rate", + value=success_rate, + ), + ModelMonitoringApplicationMetric( + name="total_input_tokens", + value=token_usage["total_input_tokens"], + ), + ModelMonitoringApplicationMetric( + name="total_output_tokens", + value=token_usage["total_output_tokens"], + ), + ModelMonitoringApplicationMetric( + name="combined_total_tokens", + value=token_usage["combined_total"], + ), + *[ModelMonitoringApplicationMetric( + name=f"run_name_counts_{run_name}", + value=count, + ) for run_name, count in run_name_counts.items()], + ] + + @staticmethod + def get_structured_runs( + monitoring_context: MonitoringApplicationContext, + labels_filter: list[str] = None, + tags_filter: list[str] = None, + run_name_filter: list[str] = None, + run_type_filter: list[str] = None, + flatten_child_runs: bool = False, + ignore_child_runs: bool = False, + ignore_errored_runs: bool = False, + ) -> tuple[list[dict], list[dict]]: + """ + Get the structured runs from the monitoring context sample dataframe. The sample dataframe contains the raw + input and output data as JSON strings - the way the MLRun tracer sends them as events to MLRun monitoring. This + function parses the JSON strings into structured dictionaries that can be used for further metrics calculations + and analysis. + + :param monitoring_context: The monitoring context containing the sample dataframe. + :param labels_filter: List of labels to filter the runs. Only runs with a label appearing in this list will + remain. If None, no filtering is applied. + :param tags_filter: List of tags to filter the runs. Only runs containing at least one tag from this list will + remain. If None, no filtering is applied. + :param run_name_filter: List of run names to filter the runs. Only runs with a name appearing in this list will + remain. If None, no filtering is applied. + :param run_type_filter: List of run types to filter the runs. Only runs with a type appearing in this list will + remain. If None, no filtering is applied. + :param flatten_child_runs: Whether to flatten child runs into the main runs list. If True, all child runs will + be extracted and added to the main runs list. If False, child runs will be kept nested within their parent + runs. + :param ignore_child_runs: Whether to ignore child runs completely. If True, child runs will be removed from the + output. If False, child runs will be processed according to the other parameters. + :param ignore_errored_runs: Whether to ignore runs that resulted in errors. If True, runs with errors will be + excluded from the output. If False, errored runs will be included. + + :returns: A list of structured run dictionaries that passed the filters and a list of samples that could not be + parsed due to errors. + """ + # Retrieve the input and output samples from the monitoring context: + samples = monitoring_context.sample_df[['input', 'output']].to_dict('records') + + # Prepare to collect structured samples: + structured_samples = [] + errored_samples = [] + + # Go over all samples: + for sample in samples: + try: + # Parse the input data into structured format: + parsed_input = orjson.loads(sample['input']) + label = parsed_input['label'] + parsed_input = parsed_input["input"]["input_data"] + # Parse the output data into structured format: + parsed_output = orjson.loads(sample['output'])["output_data"] + structured_samples.extend( + LangChainMonitoringApp._collect_run( + structured_input=parsed_input, + structured_output=parsed_output, + label=label, + labels_filter=labels_filter, + tags_filter=tags_filter, + run_name_filter=run_name_filter, + run_type_filter=run_type_filter, + flatten_child_runs=flatten_child_runs, + ignore_child_runs=ignore_child_runs, + ignore_errored_runs=ignore_errored_runs, + ) + ) + except Exception: + errored_samples.append(sample) + + return structured_samples, errored_samples + + @staticmethod + def _collect_run( + structured_input: dict, + structured_output: dict, + label: str, + child_level: int = 0, + labels_filter: list[str] = None, + tags_filter: list[str] = None, + run_name_filter: list[str] = None, + run_type_filter: list[str] = None, + flatten_child_runs: bool = False, + ignore_child_runs: bool = False, + ignore_errored_runs: bool = False, + ) -> list[dict]: + """ + Recursively collect runs from the structured input and output data, applying filters as specified. + + :param structured_input: The structured input data of the run. + :param structured_output: The structured output data of the run. + :param label: The label of the run. + :param child_level: The current child level of the run (0 for root runs). + :param labels_filter: Label filter as described in `get_structured_runs`. + :param tags_filter: Tag filter as described in `get_structured_runs`. + :param run_name_filter: Run name filter as described in `get_structured_runs`. + :param run_type_filter: Run type filter as described in `get_structured_runs`. + :param flatten_child_runs: Flag to flatten child runs as described in `get_structured_runs`. + :param ignore_child_runs: Flag to ignore child runs as described in `get_structured_runs`. + :param ignore_errored_runs: Flag to ignore errored runs as described in `get_structured_runs`. + + :returns: A list of structured run dictionaries that passed the filters. + """ + # Prepare to collect runs: + runs = [] + + # Filter by label: + if labels_filter and label not in labels_filter: + return runs + + # Handle child runs: + if "child_runs" in structured_output: + # Check if we need to ignore or flatten child runs: + if ignore_child_runs: + structured_output.pop("child_runs") + elif flatten_child_runs: + # Recursively collect child runs: + child_runs = structured_output.pop("child_runs") + flattened_runs = [] + for child_run in child_runs: + flattened_runs.extend( + LangChainMonitoringApp._collect_run( + structured_input=child_run["input_data"], + structured_output=child_run["output_data"], + label=label, + child_level=child_level + 1, + tags_filter=tags_filter, + run_name_filter=run_name_filter, + run_type_filter=run_type_filter, + flatten_child_runs=flatten_child_runs, + ignore_child_runs=ignore_child_runs, + ignore_errored_runs=ignore_errored_runs, + ) + ) + runs.extend(flattened_runs) + + # Filter by tags, run name, run type, and errors: + if tags_filter and not set(structured_input["tags"]).isdisjoint(tags_filter): + return runs + if run_name_filter and structured_input["run_name"] not in run_name_filter: + return runs + if run_type_filter and structured_input["run_type"] not in run_type_filter: + return runs + if ignore_errored_runs and structured_output.get("error", None): + return runs + + # Collect the current run: + runs.append({"label": label, "input_data": structured_input, "output_data": structured_output, + "child_level": child_level}) + return runs + + @staticmethod + def iterate_structured_runs(structured_runs: list[dict]) -> Generator[dict, None, None]: + """ + Iterates over all runs in the structured samples, including child runs. + + :param structured_runs: List of structured run samples. + + :returns: A generator yielding each run structure. + """ + # TODO: Add an option to stop at a certain child level. + for structured_run in structured_runs: + if "child_runs" in structured_run['output_data']: + for child_run in structured_run['output_data']['child_runs']: + yield from LangChainMonitoringApp.iterate_structured_runs([{ + "label": structured_run['label'], + "input_data": child_run['input_data'], + "output_data": child_run['output_data'], + "child_level": structured_run['child_level'] + 1 + }]) + yield structured_run + + @staticmethod + def count_run_names(structured_runs: list[dict]) -> dict[str, int]: + """ + Counts occurrences of each run name in the structured samples. + + :param structured_runs: List of structured run samples. + + :returns: A dictionary with run names as keys and their counts as values. + """ + # TODO: Add a nice plot artifact that will draw the bar chart for what is being used the most. + # Prepare to count run names: + run_name_counts = {} + + # Go over all the runs: + for structured_run in LangChainMonitoringApp.iterate_structured_runs(structured_runs): + run_name = structured_run['input_data']['run_name'] + if run_name in run_name_counts: + run_name_counts[run_name] += 1 + else: + run_name_counts[run_name] = 1 + + return run_name_counts + + @staticmethod + def count_token_usage(structured_runs: list[dict]) -> dict: + """ + Calculates total tokens by only counting unique 'llm' type runs. + + :param structured_runs: List of structured run samples. + + :returns: A dictionary with total input tokens, total output tokens, and combined total tokens. + """ + # TODO: Add a token count per model breakdown (a dictionary of : to token counts) + # including an artifact that will plot it nicely. Pay attention that different providers use different + # keys in the response metadata. We should implement a mapping for that so each provider will have its own + # handler that will know how to extract the relevant info out of a run. + # Prepare to count tokens: + total_input_tokens = 0 + total_output_tokens = 0 + + # Go over all the LLM typed runs: + for structured_run in LangChainMonitoringApp.iterate_structured_runs(structured_runs): + # Count only LLM type runs as chain runs may include duplicative information as they accumulate the tokens + # from the child runs: + if structured_run['input_data']['run_type'] != 'llm': + continue + # Look for the token count information: + outputs = structured_run['output_data']["outputs"] + # Newer implementations should have the metadata in the `AIMessage` kwargs under generations: + if "generations" in outputs: + for generation in outputs["generations"]: # Iterate over generations. + for sample in generation: # Iterate over the generation batch. + token_usage = sample.get("message", {}).get("kwargs", {}).get("usage_metadata", {}) + if token_usage: + total_input_tokens += ( + token_usage.get('input_tokens', 0) + or token_usage.get('prompt_tokens', 0) + ) + total_output_tokens += ( + token_usage.get('output_tokens', 0) or + token_usage.get('completion_tokens', 0) + ) + continue + # Older implementations may have the metadata under `llm_output`: + if "llm_output" in outputs: + token_usage = outputs["llm_output"].get("token_usage", {}) + if token_usage: + total_input_tokens += token_usage.get('input_tokens', 0) or token_usage.get('prompt_tokens', 0) + total_output_tokens += ( + token_usage.get('output_tokens', 0) or + token_usage.get('completion_tokens', 0) + ) + + return { + "total_input_tokens": total_input_tokens, + "total_output_tokens": total_output_tokens, + "combined_total": total_input_tokens + total_output_tokens + } + + @staticmethod + def calculate_success_rate(structured_runs: list[dict]) -> float: + """ + Calculates the success rate across all runs. + + :param structured_runs: List of structured run samples. + + :returns: Success rate as a float percentage between 0 and 1. + """ + # TODO: Add an option to see errors breakdown by kind of error and maybe an option to show which run name yielded + # most of the errors with artifacts showcasing it. + successful_count = 0 + for structured_run in structured_runs: + if 'error' not in structured_run['output_data'] or structured_run['output_data']['error'] is None: + successful_count += 1 + return successful_count / len(structured_runs) if structured_runs else 0.0 + + @staticmethod + def calculate_average_latency(structured_runs: list[dict]) -> float: + """ + Calculates the average latency across all runs. + + :param structured_runs: List of structured run samples. + + :returns: Average latency in milliseconds. + """ + # TODO: Add an option to calculate latency per run name (to know which runs are slower/faster) and then return an + # artifact showcasing it. + # Prepare to calculate average latency: + total_latency = 0.0 + count = 0 + + # Go over all the root runs: + for structured_run in structured_runs: + # Skip child runs: + if structured_run["child_level"] > 0: + continue + # Check if latency is already provided: + if "latency" in structured_run['output_data']: + total_latency += structured_run['output_data']['latency'] + count += 1 + continue + # Calculate latency from timestamps: + start_time = datetime.datetime.fromisoformat(structured_run['input_data']['start_timestamp']) + end_time = datetime.datetime.fromisoformat(structured_run['output_data']['end_timestamp']) + total_latency += (end_time - start_time).total_seconds() * 1000 # Convert to milliseconds + count += 1 + + return total_latency / count if count > 0 else 0.0 diff --git a/modules/src/langchain_mlrun/notebook_images/mlrun_ui.png b/modules/src/langchain_mlrun/notebook_images/mlrun_ui.png new file mode 100644 index 0000000000000000000000000000000000000000..9785eeae3042b9e890e70c04dd7b9f966acecbac GIT binary patch literal 85919 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 V)z4=vG!P!_g3{&ldFRah{~yTu!1MqB literal 0 HcmV?d00001 diff --git a/modules/src/langchain_mlrun/requirements.txt b/modules/src/langchain_mlrun/requirements.txt new file mode 100644 index 000000000..2597fbe12 --- /dev/null +++ b/modules/src/langchain_mlrun/requirements.txt @@ -0,0 +1,4 @@ +pytest +langchain~=1.2 +pydantic-settings~=2.12 +kafka-python~=2.3 \ No newline at end of file diff --git a/modules/src/langchain_mlrun/test_langchain_mlrun.py b/modules/src/langchain_mlrun/test_langchain_mlrun.py new file mode 100644 index 000000000..cb6ca184c --- /dev/null +++ b/modules/src/langchain_mlrun/test_langchain_mlrun.py @@ -0,0 +1,1025 @@ +# Copyright 2026 Iguazio +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +import os +from typing import Literal, TypedDict, Annotated, Sequence, Any, Callable +from concurrent.futures import ThreadPoolExecutor +from operator import add + +import pytest +from langchain_core.language_models import LanguageModelInput +from langchain_core.runnables import Runnable, RunnableLambda +from pydantic import ValidationError + +from langchain_core.prompts import ChatPromptTemplate +from langchain_core.output_parsers import StrOutputParser +from langchain_core.tracers import Run +from langchain_core.language_models.fake_chat_models import FakeListChatModel, GenericFakeChatModel +from langchain.agents import create_agent +from langchain_core.messages import AIMessage, HumanMessage +from langchain_core.tools import tool, BaseTool + +from langgraph.graph import StateGraph, START, END +from langchain_core.messages import BaseMessage +from pydantic_settings import BaseSettings, SettingsConfigDict + +from langchain_mlrun import ( + mlrun_monitoring, + MLRunTracer, + MLRunTracerSettings, + MLRunTracerClientSettings, + MLRunTracerMonitorSettings, + mlrun_monitoring_env_var, + LangChainMonitoringApp, +) + + +def _check_openai_credentials() -> bool: + """ + Check if OpenAI API key is set in environment variables. + + :return: True if OPENAI_API_KEY is set, False otherwise. + """ + return "OPENAI_API_KEY" in os.environ + + +# Import ChatOpenAI only if OpenAI credentials are available (meaning `langchain-openai` must be installed). +if _check_openai_credentials(): + from langchain_openai import ChatOpenAI + + +class _ToolEnabledFakeModel(GenericFakeChatModel): + """ + A fake chat model that supports tool binding for running agent tracing tests. + """ + + def bind_tools( + self, + tools: Sequence[ + dict[str, Any] | type | Callable | BaseTool # noqa: UP006 + ], + *, + tool_choice: str | None = None, + **kwargs: Any, + ) -> Runnable[LanguageModelInput, AIMessage]: + return self + + +#: Tag value for testing tag filtering. +_dummy_tag = "dummy_tag" + + +def _run_simple_chain() -> str: + """ + Run a simple LangChain chain that gets a fact about a topic. + """ + # Build a simple chain: prompt -> llm -> str output parser + llm = ChatOpenAI( + model="gpt-4o-mini", + tags=[_dummy_tag] + ) if _check_openai_credentials() else ( + FakeListChatModel( + responses=[ + "MLRun is an open-source orchestrator for machine learning pipelines." + ], + tags=[_dummy_tag] + ) + ) + prompt = ChatPromptTemplate.from_template("Tell me a short fact about {topic}") + chain = prompt | llm | StrOutputParser() + + # Run the chain: + response = chain.invoke({"topic": "MLRun"}) + return response + + +def _run_simple_agent(): + """ + Run a simple LangChain agent that uses two tools to get weather and stock price. + """ + # Define the tools: + @tool + def get_weather(city: str) -> str: + """Get the current weather for a specific city.""" + return f"The weather in {city} is 22°C and sunny." + + @tool + def get_stock_price(symbol: str) -> str: + """Get the current stock price for a symbol.""" + return f"The stock price for {symbol} is $150.25." + + # Define the model: + model = ChatOpenAI( + model="gpt-4o-mini", + tags=[_dummy_tag] + ) if _check_openai_credentials() else ( + _ToolEnabledFakeModel( + messages=iter( + [ + AIMessage( + content="", + tool_calls=[ + {"name": "get_weather", "args": {"city": "London"}, "id": "call_abc123"}, + {"name": "get_stock_price", "args": {"symbol": "AAPL"}, "id": "call_def456"} + ] + ), + AIMessage(content="The weather in London is 22°C and AAPL is trading at $150.25.") + ] + ), + tags=[_dummy_tag] + ) + ) + + # Create the agent: + agent = create_agent( + model=model, + tools=[get_weather, get_stock_price], + system_prompt="You are a helpful assistant with access to tools." + ) + + # Run the agent: + return agent.invoke({"messages": ["What is the weather in London and the stock price of AAPL?"]}) + + +def _run_langgraph_graph(): + """ + Run a LangGraph agent that uses reflection to correct its answer. + """ + + # Define the graph state: + class AgentState(TypedDict): + messages: Annotated[list[BaseMessage], add] + attempts: int + + # Define the model: + model = ChatOpenAI(model="gpt-4o-mini") if _check_openai_credentials() else ( + _ToolEnabledFakeModel( + messages=iter( + [ + AIMessage(content="There are 2 'r's in Strawberry."), # Mocking the failure + AIMessage(content="I stand corrected. S-t-r-a-w-b-e-r-r-y. There are 3 'r's."), # Mocking the fix + ] + ) + ) + ) + + # Define the graph nodes and router: + def call_model(state: AgentState): + response = model.invoke(state["messages"]) + return {"messages": [response], "attempts": state["attempts"] + 1} + + def reflect_node(state: AgentState): + prompt = "Wait, count the 'r's again slowly, letter by letter. Are you sure?" + return {"messages": [HumanMessage(content=prompt)]} + + def router(state: AgentState) -> Literal["reflect", END]: + # Make sure there are 2 attempts at least for an answer: + if state["attempts"] == 1: + return "reflect" + return END + + # Build the graph: + builder = StateGraph(AgentState) + builder.add_node("model", call_model) + tagged_reflect_node = RunnableLambda(reflect_node).with_config(tags=[_dummy_tag]) + builder.add_node("reflect", tagged_reflect_node) + builder.add_edge(START, "model") + builder.add_conditional_edges("model", router) + builder.add_edge("reflect", "model") + graph = builder.compile() + + # Run the graph: + return graph.invoke({"messages": [HumanMessage(content="How many 'r's in Strawberry?")], "attempts": 0}) + + +#: List of example functions to run in tests along the full (split-run enabled) expected monitor events. +_run_suites: list[tuple[Callable, int]] = [ + (_run_simple_chain, 4), + (_run_simple_agent, 9), + (_run_langgraph_graph, 9), +] + + +#: Dummy environment variables for testing. +_dummy_environment_variables = { + "LC_MLRUN_TRACER_CLIENT_V3IO_STREAM_PATH": "dummy_stream_path", + "LC_MLRUN_TRACER_CLIENT_V3IO_CONTAINER": "dummy_container", + "LC_MLRUN_TRACER_CLIENT_MODEL_ENDPOINT_NAME": "dummy_model_name", + "LC_MLRUN_TRACER_CLIENT_MODEL_ENDPOINT_UID": "dummy_model_endpoint_uid", + "LC_MLRUN_TRACER_CLIENT_SERVING_FUNCTION": "dummy_serving_function", + "LC_MLRUN_TRACER_MONITOR_DEBUG": "true", + "LC_MLRUN_TRACER_MONITOR_DEBUG_TARGET_LIST": "true", + "LC_MLRUN_TRACER_MONITOR_SPLIT_RUNS": "true", +} + + +@pytest.fixture() +def auto_mode_settings(monkeypatch): + """ + Sets the environment variables to enable mlrun monitoring in 'auto' mode. + """ + # Set environment variables for the duration of the test: + monkeypatch.setenv(mlrun_monitoring_env_var, "1") + for key, value in _dummy_environment_variables.items(): + monkeypatch.setenv(key, value) + + # Reset the singleton tracer to ensure fresh initialization: + MLRunTracer._singleton_tracer = None + MLRunTracer._initialized = False + + yield + + # Reset the singleton tracer after the test: + MLRunTracer._singleton_tracer = None + MLRunTracer._initialized = False + + +@pytest.fixture +def manual_mode_settings(): + """ + Sets the mandatory client settings and debug flag for the tests. + """ + settings = MLRunTracerSettings( + client=MLRunTracerClientSettings( + v3io_stream_path="dummy_stream_path", + v3io_container="dummy_container", + model_endpoint_name="dummy_model_name", + model_endpoint_uid="dummy_model_endpoint_uid", + serving_function="dummy_serving_function", + ), + monitor=MLRunTracerMonitorSettings( + debug=True, + debug_target_list=[], + split_runs=True, # Easier to test with split runs (filters can filter per run instead of inner events) + ), + ) + + yield settings + + +def test_settings_init_via_env_vars(monkeypatch): + """ + Test that settings are correctly initialized from environment variables. + """ + #: First, ensure that without env vars, validation fails due to missing required fields: + with pytest.raises(ValidationError): + MLRunTracerSettings() + + # Now, set the environment variables for the client settings and debug flag: + for key, value in _dummy_environment_variables.items(): + monkeypatch.setenv(key, value) + + # Ensure that settings are now correctly initialized from env vars: + settings = MLRunTracerSettings() + assert settings.client.v3io_stream_path == "dummy_stream_path" + assert settings.client.v3io_container == "dummy_container" + assert settings.client.model_endpoint_name == "dummy_model_name" + assert settings.client.model_endpoint_uid == "dummy_model_endpoint_uid" + assert settings.client.serving_function == "dummy_serving_function" + assert settings.monitor.debug is True + + +@pytest.mark.parametrize( + "test_suite", [ + # Valid case: only v3io settings provided + ( + { + "v3io_stream_path": "dummy_stream_path", + "v3io_container": "dummy_container", + "model_endpoint_name": "dummy_model_name", + "model_endpoint_uid": "dummy_model_endpoint_uid", + "serving_function": "dummy_serving_function", + }, + True, + ), + # Invalid case: partial v3io settings provided + ( + { + "v3io_stream_path": "dummy_stream_path", + "model_endpoint_name": "dummy_model_name", + "model_endpoint_uid": "dummy_model_endpoint_uid", + "serving_function": "dummy_serving_function", + }, + False, + ), + # Valid case: only kafka settings provided + ( + { + "kafka_stream_profile_name": "dummy_stream_profile_name", + "model_endpoint_name": "dummy_model_name", + "model_endpoint_uid": "dummy_model_endpoint_uid", + "serving_function": "dummy_serving_function", + }, + True, + ), + # Invalid case: partial kafka settings provided + ( + { + "kafka_linger_ms": "1000", + "model_endpoint_name": "dummy_model_name", + "model_endpoint_uid": "dummy_model_endpoint_uid", + "serving_function": "dummy_serving_function", + }, + False, + ), + # Invalid case: both v3io and kafka settings provided + ( + { + "v3io_stream_path": "dummy_stream_path", + "v3io_container": "dummy_container", + "kafka_stream_profile_name": "dummy_stream_profile_name", + "model_endpoint_name": "dummy_model_name", + "model_endpoint_uid": "dummy_model_endpoint_uid", + "serving_function": "dummy_serving_function", + }, + False, + ), + # Invalid case: both v3io and kafka settings provided (partial) + ( + { + "v3io_container": "dummy_container", + "kafka_linger_ms": "1000", + "model_endpoint_name": "dummy_model_name", + "model_endpoint_uid": "dummy_model_endpoint_uid", + "serving_function": "dummy_serving_function", + }, + False, + ), + ] +) +def test_settings_v3io_kafka_combination(test_suite: tuple[dict[str, str], bool]): + """ + Test that settings validation enforces mutual exclusivity between v3io and kafka configurations. + + :param test_suite: A tuple containing environment variable overrides and a flag indicating + whether validation should pass. + """ + settings, should_pass = test_suite + + if should_pass: + MLRunTracerClientSettings(**settings) + else: + with pytest.raises(ValidationError): + MLRunTracerClientSettings(**settings) + + +def test_auto_mode_singleton_thread_safety(auto_mode_settings): + """ + Test that MLRunTracer singleton initialization is thread-safe in 'auto' mode. + + :param auto_mode_settings: Fixture to set up 'auto' mode environment and settings. + """ + # Initialize a list to hold tracer instances created in different threads: + tracer_instances = [] + + # Function to initialize the tracer in a thread: + def _init_tracer(): + tracer = MLRunTracer() + return tracer + + # Use ThreadPoolExecutor to simulate concurrent tracer initialization: + num_threads = 50 + with ThreadPoolExecutor(max_workers=num_threads) as executor: + futures = [executor.submit(_init_tracer) for _ in range(num_threads)] + tracer_instances = [f.result() for f in futures] + + # Check if every single reference in the list is the exact same object: + unique_instances = set(tracer._uid for tracer in tracer_instances) + + assert len(tracer_instances) == num_threads, "Not all threads returned a tracer instance. Test cannot proceed." + assert len(unique_instances) == 1, ( + f"Thread-safety failure! {len(unique_instances)} different instances were created under high concurrency." + ) + assert tracer_instances[0] is MLRunTracer(), "The global access point should return the same singleton." + + +def test_manual_mode_multi_instances(manual_mode_settings: MLRunTracerSettings): + """ + Test that MLRunTracer allows multiple instances in 'manual' mode. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + """ + # Initialize a list to hold tracer instances created in different iterations: + tracer_instances = [] + + # Create multiple tracer instances: + num_instances = 50 + for _ in range(num_instances): + tracer = MLRunTracer(settings=manual_mode_settings) + tracer_instances.append(tracer) + + # Check if every single reference in the list is a different object: + unique_instances = set(tracer._uid for tracer in tracer_instances) + + assert len(tracer_instances) == num_instances, "Not all instances were created. Test cannot proceed." + assert len(unique_instances) == num_instances, ( + f"Manual mode failure! {len(unique_instances)} unique instances were created instead of {num_instances}." + ) + + +@pytest.mark.parametrize("run_suites", _run_suites) +def test_auto_mode(auto_mode_settings, run_suites: tuple[Callable, int]): + """ + Test that MLRunTracer in 'auto' mode captures debug target list after running a LangChain / LangGraph example code. + + :param auto_mode_settings: Fixture to set up 'auto' mode environment and settings. + + :param run_suites: The function to run with the expected monitored events. + """ + run_func, expected_events = run_suites + + tracer = MLRunTracer() + assert len(tracer.settings.monitor.debug_target_list) == 0 + + print(run_func()) + assert len(tracer.settings.monitor.debug_target_list) == expected_events + + +@pytest.mark.parametrize("run_suites", _run_suites) +def test_manual_mode(manual_mode_settings: MLRunTracerSettings, run_suites: tuple[Callable, int]): + """ + Test that MLRunTracer in 'auto' mode captures debug target list after running a LangChain / LangGraph example code. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + :param run_suites: The function to run with the expected monitored events. + """ + run_func, expected_events = run_suites + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + print(run_func()) + assert len(tracer.settings.monitor.debug_target_list) == expected_events + + +def test_labeling(manual_mode_settings: MLRunTracerSettings): + """ + Test that MLRunTracer in 'auto' mode captures debug target list after running a LangChain / LangGraph example code. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + """ + for i, (run_func, expected_events) in enumerate(_run_suites): + label = f"label_{i}" + manual_mode_settings.monitor.label = label + manual_mode_settings.monitor.debug_target_list.clear() + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + print(run_func()) + assert len(tracer.settings.monitor.debug_target_list) == expected_events + for event in tracer.settings.monitor.debug_target_list: + assert event["label"] == label + + +@pytest.mark.parametrize( + "run_suites", [ + run_suite + (filtered_events,) + for run_suite, filtered_events in zip(_run_suites, [1, 2, 1]) + ] +) +def test_monitor_settings_tags_filter( + manual_mode_settings: MLRunTracerSettings, + run_suites: tuple[Callable, int, int], +): + """ + Test the `tags_filter` setting of MLRunTracer. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + :param run_suites: The function to run with the expected monitored events and filtered events. + """ + run_func, expected_events, filtered_events = run_suites + + manual_mode_settings.monitor.tags_filter = [_dummy_tag] + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + print(run_func()) + assert len(tracer.settings.monitor.debug_target_list) == filtered_events + for event in tracer.settings.monitor.debug_target_list: + assert not set(manual_mode_settings.monitor.tags_filter).isdisjoint(event["input_data"]["input_data"]["tags"]) + + +@pytest.mark.parametrize( + "run_suites", [ + run_suite + (filtered_events,) + for run_suite, filtered_events in zip(_run_suites, [1, 3, 4]) + ] +) +def test_monitor_settings_name_filter( + manual_mode_settings: MLRunTracerSettings, + run_suites: tuple[Callable, int, int], +): + """ + Test the `names_filter` setting of MLRunTracer. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + :param run_suites: The function to run with the expected monitored events and filtered events. + """ + run_func, expected_events, filtered_events = run_suites + + manual_mode_settings.monitor.names_filter = ["StrOutputParser", "get_weather", "model", "router"] + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + print(run_func()) + assert len(tracer.settings.monitor.debug_target_list) == filtered_events + for event in tracer.settings.monitor.debug_target_list: + assert event["input_data"]["input_data"]["run_name"] in manual_mode_settings.monitor.names_filter + + +@pytest.mark.parametrize( + "run_suites", [ + run_suite + (filtered_events,) + for run_suite, filtered_events in zip(_run_suites, [2, 7, 9]) + ] +) +@pytest.mark.parametrize("split_runs", [True, False]) +def test_monitor_settings_run_type_filter( + manual_mode_settings: MLRunTracerSettings, + run_suites: tuple[Callable, int, int], + split_runs: bool +): + """ + Test the `run_types_filter` setting of MLRunTracer. Will also test with split runs enabled and disabled - meaning + that when disabled, if a parent run is filtered, all its child runs are also filtered by default. In the test we + made sure that the root run is always passing the filter (hence the equal one). + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + :param run_suites: The function to run with the expected monitored events and filtered events. + :param split_runs: Whether to enable split runs in the monitor settings. + """ + run_func, expected_events, filtered_events = run_suites + filtered_events = filtered_events if split_runs else 1 + + manual_mode_settings.monitor.run_types_filter = ["llm", "chain"] + manual_mode_settings.monitor.split_runs = split_runs + + def recursive_check_run_types(run: dict): + assert run["input_data"]["run_type"] in manual_mode_settings.monitor.run_types_filter + if "child_runs" in run["output_data"]: + for child_run in run["output_data"]["child_runs"]: + recursive_check_run_types(child_run) + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + print(run_func()) + assert len(tracer.settings.monitor.debug_target_list) == filtered_events + + for event in tracer.settings.monitor.debug_target_list: + event_run = { + "input_data": event["input_data"]["input_data"], + "output_data": event["output_data"]["output_data"], + } + recursive_check_run_types(run=event_run) + +@pytest.mark.parametrize("run_suites", _run_suites) +@pytest.mark.parametrize("split_runs", [True, False]) +def test_monitor_settings_full_filter( + manual_mode_settings: MLRunTracerSettings, + run_suites: tuple[Callable, int], + split_runs: bool +): + """ + Test that a complete filter (not allowing any events to pass) won't fail the tracer. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + :param run_suites: The function to run with the expected monitored events. + :param split_runs: Whether to enable split runs in the monitor settings. + """ + run_func, _ = run_suites + + manual_mode_settings.monitor.run_types_filter = ["dummy_run_type"] + manual_mode_settings.monitor.split_runs = split_runs + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + print(run_func()) + assert len(tracer.settings.monitor.debug_target_list) == 0 + + +@pytest.mark.parametrize("run_suites", _run_suites) +@pytest.mark.parametrize("split_runs", [True, False]) +@pytest.mark.parametrize("root_run_only", [True, False]) +def test_monitor_settings_split_runs_and_root_run_only( + manual_mode_settings: MLRunTracerSettings, + run_suites: tuple[Callable, int], + split_runs: bool, + root_run_only: bool, +): + """ + Test the `split_runs` setting of MLRunTracer. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + :param run_suites: The function to run with the expected monitored events. + :param split_runs: Whether to enable split runs in the monitor settings. + :param root_run_only: Whether to enable `root_run_only` in the monitor settings. + """ + run_func, expected_events = run_suites + + manual_mode_settings.monitor.split_runs = split_runs + manual_mode_settings.monitor.root_run_only = root_run_only + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + for run_iteration in range(1, 3): + print(run_func()) + if root_run_only: + assert len(tracer.settings.monitor.debug_target_list) == 1 * run_iteration + assert "child_runs" not in tracer.settings.monitor.debug_target_list[-1]["output_data"]["output_data"] + elif split_runs: + assert len(tracer.settings.monitor.debug_target_list) == expected_events * run_iteration + assert "child_runs" not in tracer.settings.monitor.debug_target_list[-1]["output_data"]["output_data"] + else: # split_runs disabled + assert len(tracer.settings.monitor.debug_target_list) == 1 * run_iteration + assert len(tracer.settings.monitor.debug_target_list[-1]["output_data"]["output_data"]["child_runs"]) != 0 + + +class _CustomRunSummarizerSettings(BaseSettings): + """ + Settings for the custom summarizer function. + """ + dummy_value: int = 21 + + model_config = SettingsConfigDict(env_prefix="TEST_CUSTOM_SUMMARIZER_SETTINGS_") + + +def _custom_run_summarizer(run: Run, settings: _CustomRunSummarizerSettings = None): + """ + A custom summarizer function for testing. + + :param run: The LangChain / LangGraph run to summarize. + :param settings: Optional settings for the summarizer. + """ + inputs = { + "run_id": run.id, + "input": run.inputs, + "from_settings": settings.dummy_value if settings else 0, + } + + def count_llm_calls(r: Run) -> int: + if not r.child_runs: + return 1 if r.run_type == "llm" else 0 + return sum(count_llm_calls(child) for child in r.child_runs) + + def count_tool_calls(r: Run) -> int: + if not r.child_runs: + return 1 if r.run_type == "tool" else 0 + return sum(count_tool_calls(child) for child in r.child_runs) + + outputs = { + "llm_calls": count_llm_calls(run), + "tool_calls": count_tool_calls(run), + "output": run.outputs + } + + yield inputs, outputs + + +@pytest.mark.parametrize("run_suites", _run_suites) +@pytest.mark.parametrize("run_summarizer_function", [ + _custom_run_summarizer, + "test_langchain_mlrun._custom_run_summarizer", +]) +@pytest.mark.parametrize("run_summarizer_settings", [ + _CustomRunSummarizerSettings(dummy_value=12), + "test_langchain_mlrun._CustomRunSummarizerSettings", + None, +]) +def test_monitor_settings_custom_run_summarizer( + manual_mode_settings: MLRunTracerSettings, + run_suites: tuple[Callable, int], + run_summarizer_function: Callable | str, + run_summarizer_settings: BaseSettings | str | None, +): + """ + Test the custom run summarizer that can be passed to MLRunTracer. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + :param run_suites: The function to run with the expected monitored events. + :param run_summarizer_function: The custom summarizer function or its import path. + :param run_summarizer_settings: The settings for the custom summarizer or its import path. + """ + run_func, _ = run_suites + manual_mode_settings.monitor.run_summarizer_function = run_summarizer_function + manual_mode_settings.monitor.run_summarizer_settings = run_summarizer_settings + dummy_value_for_settings_from_env = 26 + os.environ["TEST_CUSTOM_SUMMARIZER_SETTINGS_DUMMY_VALUE"] = str(dummy_value_for_settings_from_env) + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + print(run_func()) + assert len(tracer.settings.monitor.debug_target_list) == 1 + + event = tracer.settings.monitor.debug_target_list[0] + if run_summarizer_settings: + if isinstance(run_summarizer_settings, str): + assert event["input_data"]["input_data"]["from_settings"] == dummy_value_for_settings_from_env + else: + assert event["input_data"]["input_data"]["from_settings"] == run_summarizer_settings.dummy_value + else: + assert event["input_data"]["input_data"]["from_settings"] == 0 + + +def test_monitor_settings_include_errors_field_presence(manual_mode_settings: MLRunTracerSettings): + """ + Test that when `include_errors` is True, the error field is present in outputs. + When `include_errors` is False, the error field is not added to outputs. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + """ + # Run with include_errors=True (default) and verify error field is present: + manual_mode_settings.monitor.include_errors = True + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + _run_simple_chain() + assert len(tracer.settings.monitor.debug_target_list) > 0 + + for event in tracer.settings.monitor.debug_target_list: + output_data = event["output_data"]["output_data"] + assert "error" in output_data, "error field should be present when include_errors is True" + + # Now run with include_errors=False and verify error field is excluded: + manual_mode_settings.monitor.include_errors = False + manual_mode_settings.monitor.debug_target_list.clear() + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + _run_simple_chain() + assert len(tracer.settings.monitor.debug_target_list) > 0 + + for event in tracer.settings.monitor.debug_target_list: + output_data = event["output_data"]["output_data"] + assert "error" not in output_data, "error field should be excluded when include_errors is False" + + +def test_monitor_settings_include_full_run(manual_mode_settings: MLRunTracerSettings): + """ + Test that when `include_full_run` is True, the complete serialized run is included in outputs. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + """ + manual_mode_settings.monitor.include_full_run = True + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + _run_simple_chain() + + assert len(tracer.settings.monitor.debug_target_list) > 0 + + for event in tracer.settings.monitor.debug_target_list: + output_data = event["output_data"]["output_data"] + assert "full_run" in output_data, "full_run should be included in outputs when include_full_run is True" + # Verify the full_run contains expected run structure: + assert "inputs" in output_data["full_run"] + assert "outputs" in output_data["full_run"] + + +def test_monitor_settings_include_metadata(manual_mode_settings: MLRunTracerSettings): + """ + Test that when `include_metadata` is False, metadata is excluded from inputs. + + Note: The fake models used in tests don't produce runs with metadata, so we can only + verify the "exclude" behavior. The code only adds metadata if the run actually contains it. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + """ + # Run with include_metadata=False and verify metadata is excluded: + manual_mode_settings.monitor.include_metadata = False + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + _run_simple_chain() + assert len(tracer.settings.monitor.debug_target_list) > 0 + + # Check that metadata is not present in inputs: + for event in tracer.settings.monitor.debug_target_list: + input_data = event["input_data"]["input_data"] + assert "metadata" not in input_data, "metadata should be excluded when include_metadata is False" + + +def test_monitor_settings_include_latency(manual_mode_settings: MLRunTracerSettings): + """ + Test that when `include_latency` is False, latency is excluded from outputs. + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + """ + manual_mode_settings.monitor.include_latency = False + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + _run_simple_chain() + assert len(tracer.settings.monitor.debug_target_list) > 0 + + for event in tracer.settings.monitor.debug_target_list: + assert "latency" not in event["output_data"]["output_data"], \ + "latency should be excluded when include_latency is False" + + +def test_import_from_module_path_errors(): + """ + Test that `_import_from_module_path` raises appropriate errors for invalid paths. + """ + # Test ValueError for path without a dot: + with pytest.raises(ValueError) as exc_info: + MLRunTracer._import_from_module_path("no_dot_path") + assert "must have at least one '.'" in str(exc_info.value) + + # Test ImportError for non-existent module: + with pytest.raises(ImportError) as exc_info: + MLRunTracer._import_from_module_path("nonexistent_module_xyz.SomeClass") + assert "Could not import" in str(exc_info.value) + + # Test AttributeError for non-existent attribute in existing module: + with pytest.raises(AttributeError) as exc_info: + MLRunTracer._import_from_module_path("os.nonexistent_attribute_xyz") + assert "Could not import" in str(exc_info.value) + + +#: Sample structured runs for testing LangChainMonitoringApp methods. +_sample_structured_runs = [ + { + "label": "test_label", + "child_level": 0, + "input_data": { + "run_name": "RunnableSequence", + "run_type": "chain", + "tags": ["tag1"], + "inputs": {"topic": "MLRun"}, + "start_timestamp": "2024-01-01T10:00:00+00:00", + }, + "output_data": { + "outputs": {"result": "test output"}, + "end_timestamp": "2024-01-01T10:00:01+00:00", + "error": None, + "child_runs": [ + { + "input_data": { + "run_name": "FakeListChatModel", + "run_type": "llm", + "tags": ["tag2"], + "inputs": {"prompt": "test"}, + "start_timestamp": "2024-01-01T10:00:00.100+00:00", + }, + "output_data": { + "outputs": { + "generations": [[{ + "message": { + "kwargs": { + "usage_metadata": { + "input_tokens": 10, + "output_tokens": 20, + } + } + } + }]] + }, + "end_timestamp": "2024-01-01T10:00:00.500+00:00", + "error": None, + }, + }, + ], + }, + }, + { + "label": "test_label", + "child_level": 0, + "input_data": { + "run_name": "SimpleAgent", + "run_type": "chain", + "tags": ["tag1"], + "inputs": {"query": "test query"}, + "start_timestamp": "2024-01-01T10:00:02+00:00", + }, + "output_data": { + "outputs": {"result": "agent output"}, + "end_timestamp": "2024-01-01T10:00:04+00:00", + "error": "SomeError: something went wrong", + }, + }, +] + + +def test_langchain_monitoring_app_iterate_structured_runs(): + """ + Test that `iterate_structured_runs` yields all runs including nested child runs. + """ + # Iterate over all runs: + all_runs = list(LangChainMonitoringApp.iterate_structured_runs(_sample_structured_runs)) + + # Should yield parent runs and child runs: + # - First sample: 1 parent + 1 child = 2 runs + # - Second sample: 1 parent = 1 run + # Total: 3 runs + assert len(all_runs) == 3 + + # Verify run names are as expected: + run_names = [r["input_data"]["run_name"] for r in all_runs] + assert "RunnableSequence" in run_names + assert "FakeListChatModel" in run_names + assert "SimpleAgent" in run_names + + +def test_langchain_monitoring_app_count_run_names(): + """ + Test that `count_run_names` correctly counts occurrences of each run name. + """ + counts = LangChainMonitoringApp.count_run_names(_sample_structured_runs) + + assert counts["RunnableSequence"] == 1 + assert counts["FakeListChatModel"] == 1 + assert counts["SimpleAgent"] == 1 + + +def test_langchain_monitoring_app_count_token_usage(): + """ + Test that `count_token_usage` correctly calculates total tokens from LLM runs. + """ + token_usage = LangChainMonitoringApp.count_token_usage(_sample_structured_runs) + + assert token_usage["total_input_tokens"] == 10 + assert token_usage["total_output_tokens"] == 20 + assert token_usage["combined_total"] == 30 + + +def test_langchain_monitoring_app_calculate_success_rate(): + """ + Test that `calculate_success_rate` returns the correct percentage of successful runs. + """ + success_rate = LangChainMonitoringApp.calculate_success_rate(_sample_structured_runs) + + # First run has no error, second run has error: + # Success rate should be 1/2 = 0.5 + assert success_rate == 0.5 + + # Test with empty list: + empty_rate = LangChainMonitoringApp.calculate_success_rate([]) + assert empty_rate == 0.0 + + # Test with all successful runs: + successful_runs = [_sample_structured_runs[0]] # Only the first run which has no error + all_success_rate = LangChainMonitoringApp.calculate_success_rate(successful_runs) + assert all_success_rate == 1.0 + + +def test_langchain_monitoring_app_calculate_average_latency(): + """ + Test that `calculate_average_latency` returns the correct average latency across root runs. + """ + # Calculate average latency: + avg_latency = LangChainMonitoringApp.calculate_average_latency(_sample_structured_runs) + + # First run: 10:00:00 to 10:00:01 = 1000ms + # Second run: 10:00:02 to 10:00:04 = 2000ms + # Average: (1000 + 2000) / 2 = 1500ms + assert avg_latency == 1500.0 + + # Test with empty list: + empty_latency = LangChainMonitoringApp.calculate_average_latency([]) + assert empty_latency == 0.0 + + +def test_langchain_monitoring_app_calculate_average_latency_skips_child_runs(): + """ + Test that `calculate_average_latency` skips child runs (only calculates for root runs). + """ + # Create a sample with a child run that has child_level > 0: + runs_with_child = [ + { + "label": "test", + "child_level": 0, + "input_data": {"start_timestamp": "2024-01-01T10:00:00+00:00"}, + "output_data": {"end_timestamp": "2024-01-01T10:00:01+00:00"}, + }, + { + "label": "test", + "child_level": 1, # This is a child run, should be skipped + "input_data": {"start_timestamp": "2024-01-01T10:00:00+00:00"}, + "output_data": {"end_timestamp": "2024-01-01T10:00:10+00:00"}, # 10 seconds - would skew average + }, + ] + + # Calculate average latency: + avg_latency = LangChainMonitoringApp.calculate_average_latency(runs_with_child) + + # Should only consider the root run (1000ms), not the child run: + assert avg_latency == 1000.0 + + +def test_debug_mode_stdout(manual_mode_settings: MLRunTracerSettings, capsys): + """ + Test that debug mode prints to stdout when `debug_target_list` is not set (is False). + + :param manual_mode_settings: Fixture to set up 'manual' mode environment and settings. + :param capsys: Pytest fixture to capture stdout/stderr. + """ + # Set debug mode with debug_target_list=False (should print to stdout): + manual_mode_settings.monitor.debug = True + manual_mode_settings.monitor.debug_target_list = False + + with mlrun_monitoring(settings=manual_mode_settings) as tracer: + _run_simple_chain() + + # Capture stdout: + captured = capsys.readouterr() + + # Verify that JSON output was printed to stdout: + assert "event_id" in captured.out, "Event should be printed to stdout when debug_target_list is False" + assert "input_data" in captured.out + assert "output_data" in captured.out From 99924a6bd6ca8031d908ae6a50464083c9262b33 Mon Sep 17 00:00:00 2001 From: Omer Mimon <81911093+omermaim@users.noreply.github.com> Date: Wed, 11 Feb 2026 12:48:12 +0200 Subject: [PATCH 3/3] [onnx_utils] updated pytorch due to security vulnerability (#968) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Update onnx_utils dependencies and improve test robustness - Upgrade PyTorch (2.6.0 → 2.8.0) and TorchVision (0.21.0 → 0.23.0) for better compatibility and performance - Update MLRun version requirement to 1.10.0 in item.yaml - Bump function version to 1.4.0 Test improvements: - Add environment variable validation (MLRUN_DBPATH, MLRUN_ARTIFACT_PATH) - Add conditional test skipping based on tf2onnx availability - Fix cleanup function to properly remove test artifacts (model.pt, model_modules_map.json, onnx_model.onnx, etc.) - Update deprecated artifact_path parameter to output_path - Add explicit project context to all MLRun function calls - Fix PyTorch test artifact path construction * Add conftest fixture for test environment and update notebook to PyTorch demo - Centralize test setup/cleanup in conftest autouse fixture - Rewrite notebook demo from Keras to a working PyTorch MobileNetV2 example * deleted iguazio credentials * Remove conftest.py and inline fixtures into test_onnx_utils.py Move onnx_project and test_environment fixtures directly into the test file to reduce unnecessary indirection for a single test module. --- functions/src/onnx_utils/function.yaml | 74 +- functions/src/onnx_utils/item.yaml | 8 +- functions/src/onnx_utils/onnx_utils.ipynb | 961 ++++++++++++++++++-- functions/src/onnx_utils/requirements.txt | 7 +- functions/src/onnx_utils/test_onnx_utils.py | 219 +++-- 5 files changed, 1054 insertions(+), 215 deletions(-) diff --git a/functions/src/onnx_utils/function.yaml b/functions/src/onnx_utils/function.yaml index 05a0f0bc2..091002cdc 100644 --- a/functions/src/onnx_utils/function.yaml +++ b/functions/src/onnx_utils/function.yaml @@ -1,39 +1,13 @@ -kind: job metadata: + name: onnx-utils + tag: '' categories: - utilities - deep-learning - name: onnx-utils - tag: '' -verbose: false +kind: job spec: - build: - code_origin: '' - base_image: mlrun/mlrun - origin_filename: '' - functionSourceCode: # Copyright 2019 Iguazio
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
from typing import Any, Callable, Dict, List, Tuple

import mlrun


class _ToONNXConversions:
    """
    An ONNX conversion functions library class.
    """

    @staticmethod
    def tf_keras_to_onnx(
        model_handler,
        onnx_model_name: str = None,
        optimize_model: bool = True,
        input_signature: List[Tuple[Tuple[int], str]] = None,
    ):
        """
        Convert a TF.Keras model to an ONNX model and log it back to MLRun as a new model object.

        :param model_handler:   An initialized TFKerasModelHandler with a loaded model to convert to ONNX.
        :param onnx_model_name: The name to use to log the converted ONNX model. If not given, the given `model_name`
                                will be used with an additional suffix `_onnx`. Defaulted to None.
        :param optimize_model:  Whether or not to optimize the ONNX model using 'onnxoptimizer' before saving the model.
                                Defaulted to True.
        :param input_signature: A list of the input layers shape and data type properties. Expected to receive a list
                                where each element is an input layer tuple. An input layer tuple is a tuple of:
                                [0] = Layer's shape, a tuple of integers.
                                [1] = Layer's data type, a mlrun.data_types.ValueType string.
                                If None, the input signature will be tried to be read from the model artifact. Defaulted
                                to None.
        """
        # Import the framework and handler:
        import tensorflow as tf
        from mlrun.frameworks.tf_keras import TFKerasUtils

        # Check the given 'input_signature' parameter:
        if input_signature is None:
            # Read the inputs from the model:
            try:
                model_handler.read_inputs_from_model()
            except Exception as error:
                raise mlrun.errors.MLRunRuntimeError(
                    f"Please provide the 'input_signature' parameter. The function tried reading the input layers "
                    f"information automatically but failed with the following error: {error}"
                )
        else:
            # Parse the 'input_signature' parameter:
            input_signature = [
                tf.TensorSpec(
                    shape=shape,
                    dtype=TFKerasUtils.convert_value_type_to_tf_dtype(
                        value_type=value_type
                    ),
                )
                for (shape, value_type) in input_signature
            ]

        # Convert to ONNX:
        model_handler.to_onnx(
            model_name=onnx_model_name,
            input_signature=input_signature,
            optimize=optimize_model,
        )

    @staticmethod
    def pytorch_to_onnx(
        model_handler,
        onnx_model_name: str = None,
        optimize_model: bool = True,
        input_signature: List[Tuple[Tuple[int, ...], str]] = None,
        input_layers_names: List[str] = None,
        output_layers_names: List[str] = None,
        dynamic_axes: Dict[str, Dict[int, str]] = None,
        is_batched: bool = True,
    ):
        """
        Convert a PyTorch model to an ONNX model and log it back to MLRun as a new model object.

        :param model_handler:       An initialized PyTorchModelHandler with a loaded model to convert to ONNX.
        :param onnx_model_name:     The name to use to log the converted ONNX model. If not given, the given
                                    `model_name` will be used with an additional suffix `_onnx`. Defaulted to None.
        :param optimize_model:      Whether or not to optimize the ONNX model using 'onnxoptimizer' before saving the
                                    model. Defaulted to True.
        :param input_signature:     A list of the input layers shape and data type properties. Expected to receive a
                                    list where each element is an input layer tuple. An input layer tuple is a tuple of:
                                    [0] = Layer's shape, a tuple of integers.
                                    [1] = Layer's data type, a mlrun.data_types.ValueType string.
                                    If None, the input signature will be tried to be read from the model artifact.
                                    Defaulted to None.
        :param input_layers_names:  List of names to assign to the input nodes of the graph in order. All of the other
                                    parameters (inner layers) can be set as well by passing additional names in the
                                    list. The order is by the order of the parameters in the model. If None, the inputs
                                    will be read from the handler's inputs. If its also None, it is defaulted to:
                                    "input_0", "input_1", ...
        :param output_layers_names: List of names to assign to the output nodes of the graph in order. If None, the
                                    outputs will be read from the handler's outputs. If its also None, it is defaulted
                                    to: "output_0" (for multiple outputs, this parameter must be provided).
        :param dynamic_axes:        If part of the input / output shape is dynamic, like (batch_size, 3, 32, 32) you can
                                    specify it by giving a dynamic axis to the input / output layer by its name as
                                    follows: {
                                        "input layer name": {0: "batch_size"},
                                        "output layer name": {0: "batch_size"},
                                    }
                                    If provided, the 'is_batched' flag will be ignored. Defaulted to None.
        :param is_batched:          Whether to include a batch size as the first axis in every input and output layer.
                                    Defaulted to True. Will be ignored if 'dynamic_axes' is provided.
        """
        # Import the framework and handler:
        import torch
        from mlrun.frameworks.pytorch import PyTorchUtils

        # Parse the 'input_signature' parameter:
        if input_signature is not None:
            input_signature = tuple(
                [
                    torch.zeros(
                        size=shape,
                        dtype=PyTorchUtils.convert_value_type_to_torch_dtype(
                            value_type=value_type
                        ),
                    )
                    for (shape, value_type) in input_signature
                ]
            )

        # Convert to ONNX:
        model_handler.to_onnx(
            model_name=onnx_model_name,
            input_sample=input_signature,
            optimize=optimize_model,
            input_layers_names=input_layers_names,
            output_layers_names=output_layers_names,
            dynamic_axes=dynamic_axes,
            is_batched=is_batched,
        )


# Map for getting the conversion function according to the provided framework:
_CONVERSION_MAP = {
    "tensorflow.keras": _ToONNXConversions.tf_keras_to_onnx,
    "torch": _ToONNXConversions.pytorch_to_onnx,
}  # type: Dict[str, Callable]


def to_onnx(
    context: mlrun.MLClientCtx,
    model_path: str,
    load_model_kwargs: dict = None,
    onnx_model_name: str = None,
    optimize_model: bool = True,
    framework_kwargs: Dict[str, Any] = None,
):
    """
    Convert the given model to an ONNX model.

    :param context:           The MLRun function execution context
    :param model_path:        The model path store object.
    :param load_model_kwargs: Keyword arguments to pass to the `AutoMLRun.load_model` method.
    :param onnx_model_name:   The name to use to log the converted ONNX model. If not given, the given `model_name` will
                              be used with an additional suffix `_onnx`. Defaulted to None.
    :param optimize_model:    Whether to optimize the ONNX model using 'onnxoptimizer' before saving the model.
                              Defaulted to True.
    :param framework_kwargs:  Additional arguments each framework may require to convert to ONNX. To get the doc string
                              of the desired framework onnx conversion function, pass "help".
    """
    from mlrun.frameworks.auto_mlrun.auto_mlrun import AutoMLRun

    # Get a model handler of the required framework:
    load_model_kwargs = load_model_kwargs or {}
    model_handler = AutoMLRun.load_model(
        model_path=model_path, context=context, **load_model_kwargs
    )

    # Get the model's framework:
    framework = model_handler.FRAMEWORK_NAME

    # Use the conversion map to get the specific framework to onnx conversion:
    if framework not in _CONVERSION_MAP:
        raise mlrun.errors.MLRunInvalidArgumentError(
            f"The following framework: '{framework}', has no ONNX conversion."
        )
    conversion_function = _CONVERSION_MAP[framework]

    # Check if needed to print the function's doc string ("help" is passed):
    if framework_kwargs == "help":
        print(conversion_function.__doc__)
        return

    # Set the default empty framework kwargs if needed:
    if framework_kwargs is None:
        framework_kwargs = {}

    # Run the conversion:
    try:
        conversion_function(
            model_handler=model_handler,
            onnx_model_name=onnx_model_name,
            optimize_model=optimize_model,
            **framework_kwargs,
        )
    except TypeError as exception:
        raise mlrun.errors.MLRunInvalidArgumentError(
            f"ERROR: A TypeError exception was raised during the conversion:\n{exception}. "
            f"Please read the {framework} framework conversion function doc string by passing 'help' in the "
            f"'framework_kwargs' dictionary parameter."
        )


def optimize(
    context: mlrun.MLClientCtx,
    model_path: str,
    handler_init_kwargs: dict = None,
    optimizations: List[str] = None,
    fixed_point: bool = False,
    optimized_model_name: str = None,
):
    """
    Optimize the given ONNX model.

    :param context:              The MLRun function execution context.
    :param model_path:           Path to the ONNX model object.
    :param handler_init_kwargs:  Keyword arguments to pass to the `ONNXModelHandler` init method preloading.
    :param optimizations:        List of possible optimizations. To see what optimizations are available, pass "help".
                                 If None, all the optimizations will be used. Defaulted to None.
    :param fixed_point:          Optimize the weights using fixed point. Defaulted to False.
    :param optimized_model_name: The name of the optimized model. If None, the original model will be overridden.
                                 Defaulted to None.
    """
    # Import the model handler:
    import onnxoptimizer
    from mlrun.frameworks.onnx import ONNXModelHandler

    # Check if needed to print the available optimizations ("help" is passed):
    if optimizations == "help":
        available_passes = "\n* ".join(onnxoptimizer.get_available_passes())
        print(f"The available optimizations are:\n* {available_passes}")
        return

    # Create the model handler:
    handler_init_kwargs = handler_init_kwargs or {}
    model_handler = ONNXModelHandler(
        model_path=model_path, context=context, **handler_init_kwargs
    )

    # Load the ONNX model:
    model_handler.load()

    # Optimize the model using the given configurations:
    model_handler.optimize(optimizations=optimizations, fixed_point=fixed_point)

    # Rename if needed:
    if optimized_model_name is not None:
        model_handler.set_model_name(model_name=optimized_model_name)

    # Log the optimized model:
    model_handler.log()
 - requirements: - - tqdm~=4.67.1 - - tensorflow~=2.19.0 - - tf_keras~=2.19.0 - - torch~=2.6.0 - - torchvision~=0.21.0 - - onnx~=1.17.0 - - onnxruntime~=1.19.2 - - onnxoptimizer~=0.3.13 - - onnxmltools~=1.13.0 - - tf2onnx~=1.16.1 - - plotly~=5.23 - with_mlrun: false - auto_build: true - disable_auto_mount: false - description: ONNX intigration in MLRun, some utils functions for the ONNX framework, - optimizing and converting models from different framework to ONNX using MLRun. - image: '' entry_points: tf_keras_to_onnx: - doc: Convert a TF.Keras model to an ONNX model and log it back to MLRun as a - new model object. name: tf_keras_to_onnx parameters: - name: model_handler @@ -58,12 +32,12 @@ spec: data type, a mlrun.data_types.ValueType string. If None, the input signature will be tried to be read from the model artifact. Defaulted to None.' default: null + doc: Convert a TF.Keras model to an ONNX model and log it back to MLRun as a + new model object. + lineno: 26 has_varargs: false has_kwargs: false - lineno: 26 pytorch_to_onnx: - doc: Convert a PyTorch model to an ONNX model and log it back to MLRun as a - new model object. name: pytorch_to_onnx parameters: - name: model_handler @@ -116,11 +90,12 @@ spec: doc: Whether to include a batch size as the first axis in every input and output layer. Defaulted to True. Will be ignored if 'dynamic_axes' is provided. default: true + doc: Convert a PyTorch model to an ONNX model and log it back to MLRun as a + new model object. + lineno: 81 has_varargs: false has_kwargs: false - lineno: 81 to_onnx: - doc: Convert the given model to an ONNX model. name: to_onnx parameters: - name: context @@ -150,11 +125,11 @@ spec: get the doc string of the desired framework onnx conversion function, pass "help". default: null + doc: Convert the given model to an ONNX model. + lineno: 160 has_varargs: false has_kwargs: false - lineno: 160 optimize: - doc: Optimize the given ONNX model. name: optimize parameters: - name: context @@ -181,9 +156,34 @@ spec: doc: The name of the optimized model. If None, the original model will be overridden. Defaulted to None. default: null + doc: Optimize the given ONNX model. + lineno: 224 has_varargs: false has_kwargs: false - lineno: 224 + image: '' default_handler: to_onnx allow_empty_resources: true command: '' + disable_auto_mount: false + description: ONNX intigration in MLRun, some utils functions for the ONNX framework, + optimizing and converting models from different framework to ONNX using MLRun. + build: + functionSourceCode: # Copyright 2019 Iguazio
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
from typing import Any, Callable, Dict, List, Tuple

import mlrun


class _ToONNXConversions:
    """
    An ONNX conversion functions library class.
    """

    @staticmethod
    def tf_keras_to_onnx(
        model_handler,
        onnx_model_name: str = None,
        optimize_model: bool = True,
        input_signature: List[Tuple[Tuple[int], str]] = None,
    ):
        """
        Convert a TF.Keras model to an ONNX model and log it back to MLRun as a new model object.

        :param model_handler:   An initialized TFKerasModelHandler with a loaded model to convert to ONNX.
        :param onnx_model_name: The name to use to log the converted ONNX model. If not given, the given `model_name`
                                will be used with an additional suffix `_onnx`. Defaulted to None.
        :param optimize_model:  Whether or not to optimize the ONNX model using 'onnxoptimizer' before saving the model.
                                Defaulted to True.
        :param input_signature: A list of the input layers shape and data type properties. Expected to receive a list
                                where each element is an input layer tuple. An input layer tuple is a tuple of:
                                [0] = Layer's shape, a tuple of integers.
                                [1] = Layer's data type, a mlrun.data_types.ValueType string.
                                If None, the input signature will be tried to be read from the model artifact. Defaulted
                                to None.
        """
        # Import the framework and handler:
        import tensorflow as tf
        from mlrun.frameworks.tf_keras import TFKerasUtils

        # Check the given 'input_signature' parameter:
        if input_signature is None:
            # Read the inputs from the model:
            try:
                model_handler.read_inputs_from_model()
            except Exception as error:
                raise mlrun.errors.MLRunRuntimeError(
                    f"Please provide the 'input_signature' parameter. The function tried reading the input layers "
                    f"information automatically but failed with the following error: {error}"
                )
        else:
            # Parse the 'input_signature' parameter:
            input_signature = [
                tf.TensorSpec(
                    shape=shape,
                    dtype=TFKerasUtils.convert_value_type_to_tf_dtype(
                        value_type=value_type
                    ),
                )
                for (shape, value_type) in input_signature
            ]

        # Convert to ONNX:
        model_handler.to_onnx(
            model_name=onnx_model_name,
            input_signature=input_signature,
            optimize=optimize_model,
        )

    @staticmethod
    def pytorch_to_onnx(
        model_handler,
        onnx_model_name: str = None,
        optimize_model: bool = True,
        input_signature: List[Tuple[Tuple[int, ...], str]] = None,
        input_layers_names: List[str] = None,
        output_layers_names: List[str] = None,
        dynamic_axes: Dict[str, Dict[int, str]] = None,
        is_batched: bool = True,
    ):
        """
        Convert a PyTorch model to an ONNX model and log it back to MLRun as a new model object.

        :param model_handler:       An initialized PyTorchModelHandler with a loaded model to convert to ONNX.
        :param onnx_model_name:     The name to use to log the converted ONNX model. If not given, the given
                                    `model_name` will be used with an additional suffix `_onnx`. Defaulted to None.
        :param optimize_model:      Whether or not to optimize the ONNX model using 'onnxoptimizer' before saving the
                                    model. Defaulted to True.
        :param input_signature:     A list of the input layers shape and data type properties. Expected to receive a
                                    list where each element is an input layer tuple. An input layer tuple is a tuple of:
                                    [0] = Layer's shape, a tuple of integers.
                                    [1] = Layer's data type, a mlrun.data_types.ValueType string.
                                    If None, the input signature will be tried to be read from the model artifact.
                                    Defaulted to None.
        :param input_layers_names:  List of names to assign to the input nodes of the graph in order. All of the other
                                    parameters (inner layers) can be set as well by passing additional names in the
                                    list. The order is by the order of the parameters in the model. If None, the inputs
                                    will be read from the handler's inputs. If its also None, it is defaulted to:
                                    "input_0", "input_1", ...
        :param output_layers_names: List of names to assign to the output nodes of the graph in order. If None, the
                                    outputs will be read from the handler's outputs. If its also None, it is defaulted
                                    to: "output_0" (for multiple outputs, this parameter must be provided).
        :param dynamic_axes:        If part of the input / output shape is dynamic, like (batch_size, 3, 32, 32) you can
                                    specify it by giving a dynamic axis to the input / output layer by its name as
                                    follows: {
                                        "input layer name": {0: "batch_size"},
                                        "output layer name": {0: "batch_size"},
                                    }
                                    If provided, the 'is_batched' flag will be ignored. Defaulted to None.
        :param is_batched:          Whether to include a batch size as the first axis in every input and output layer.
                                    Defaulted to True. Will be ignored if 'dynamic_axes' is provided.
        """
        # Import the framework and handler:
        import torch
        from mlrun.frameworks.pytorch import PyTorchUtils

        # Parse the 'input_signature' parameter:
        if input_signature is not None:
            input_signature = tuple(
                [
                    torch.zeros(
                        size=shape,
                        dtype=PyTorchUtils.convert_value_type_to_torch_dtype(
                            value_type=value_type
                        ),
                    )
                    for (shape, value_type) in input_signature
                ]
            )

        # Convert to ONNX:
        model_handler.to_onnx(
            model_name=onnx_model_name,
            input_sample=input_signature,
            optimize=optimize_model,
            input_layers_names=input_layers_names,
            output_layers_names=output_layers_names,
            dynamic_axes=dynamic_axes,
            is_batched=is_batched,
        )


# Map for getting the conversion function according to the provided framework:
_CONVERSION_MAP = {
    "tensorflow.keras": _ToONNXConversions.tf_keras_to_onnx,
    "torch": _ToONNXConversions.pytorch_to_onnx,
}  # type: Dict[str, Callable]


def to_onnx(
    context: mlrun.MLClientCtx,
    model_path: str,
    load_model_kwargs: dict = None,
    onnx_model_name: str = None,
    optimize_model: bool = True,
    framework_kwargs: Dict[str, Any] = None,
):
    """
    Convert the given model to an ONNX model.

    :param context:           The MLRun function execution context
    :param model_path:        The model path store object.
    :param load_model_kwargs: Keyword arguments to pass to the `AutoMLRun.load_model` method.
    :param onnx_model_name:   The name to use to log the converted ONNX model. If not given, the given `model_name` will
                              be used with an additional suffix `_onnx`. Defaulted to None.
    :param optimize_model:    Whether to optimize the ONNX model using 'onnxoptimizer' before saving the model.
                              Defaulted to True.
    :param framework_kwargs:  Additional arguments each framework may require to convert to ONNX. To get the doc string
                              of the desired framework onnx conversion function, pass "help".
    """
    from mlrun.frameworks.auto_mlrun.auto_mlrun import AutoMLRun

    # Get a model handler of the required framework:
    load_model_kwargs = load_model_kwargs or {}
    model_handler = AutoMLRun.load_model(
        model_path=model_path, context=context, **load_model_kwargs
    )

    # Get the model's framework:
    framework = model_handler.FRAMEWORK_NAME

    # Use the conversion map to get the specific framework to onnx conversion:
    if framework not in _CONVERSION_MAP:
        raise mlrun.errors.MLRunInvalidArgumentError(
            f"The following framework: '{framework}', has no ONNX conversion."
        )
    conversion_function = _CONVERSION_MAP[framework]

    # Check if needed to print the function's doc string ("help" is passed):
    if framework_kwargs == "help":
        print(conversion_function.__doc__)
        return

    # Set the default empty framework kwargs if needed:
    if framework_kwargs is None:
        framework_kwargs = {}

    # Run the conversion:
    try:
        conversion_function(
            model_handler=model_handler,
            onnx_model_name=onnx_model_name,
            optimize_model=optimize_model,
            **framework_kwargs,
        )
    except TypeError as exception:
        raise mlrun.errors.MLRunInvalidArgumentError(
            f"ERROR: A TypeError exception was raised during the conversion:\n{exception}. "
            f"Please read the {framework} framework conversion function doc string by passing 'help' in the "
            f"'framework_kwargs' dictionary parameter."
        )


def optimize(
    context: mlrun.MLClientCtx,
    model_path: str,
    handler_init_kwargs: dict = None,
    optimizations: List[str] = None,
    fixed_point: bool = False,
    optimized_model_name: str = None,
):
    """
    Optimize the given ONNX model.

    :param context:              The MLRun function execution context.
    :param model_path:           Path to the ONNX model object.
    :param handler_init_kwargs:  Keyword arguments to pass to the `ONNXModelHandler` init method preloading.
    :param optimizations:        List of possible optimizations. To see what optimizations are available, pass "help".
                                 If None, all the optimizations will be used. Defaulted to None.
    :param fixed_point:          Optimize the weights using fixed point. Defaulted to False.
    :param optimized_model_name: The name of the optimized model. If None, the original model will be overridden.
                                 Defaulted to None.
    """
    # Import the model handler:
    import onnxoptimizer
    from mlrun.frameworks.onnx import ONNXModelHandler

    # Check if needed to print the available optimizations ("help" is passed):
    if optimizations == "help":
        available_passes = "\n* ".join(onnxoptimizer.get_available_passes())
        print(f"The available optimizations are:\n* {available_passes}")
        return

    # Create the model handler:
    handler_init_kwargs = handler_init_kwargs or {}
    model_handler = ONNXModelHandler(
        model_path=model_path, context=context, **handler_init_kwargs
    )

    # Load the ONNX model:
    model_handler.load()

    # Optimize the model using the given configurations:
    model_handler.optimize(optimizations=optimizations, fixed_point=fixed_point)

    # Rename if needed:
    if optimized_model_name is not None:
        model_handler.set_model_name(model_name=optimized_model_name)

    # Log the optimized model:
    model_handler.log()
 + base_image: mlrun/mlrun + with_mlrun: false + auto_build: true + requirements: + - tqdm~=4.67.1 + - tensorflow~=2.19.0 + - tf_keras~=2.19.0 + - torch~=2.8.0 + - torchvision~=0.23.0 + - onnx~=1.17.0 + - onnxruntime~=1.19.2 + - onnxoptimizer~=0.3.13 + - onnxmltools~=1.13.0 + - tf2onnx~=1.16.1 + - plotly~=5.23 + origin_filename: '' + code_origin: '' +verbose: false diff --git a/functions/src/onnx_utils/item.yaml b/functions/src/onnx_utils/item.yaml index 803bd2599..5f129389f 100644 --- a/functions/src/onnx_utils/item.yaml +++ b/functions/src/onnx_utils/item.yaml @@ -13,7 +13,7 @@ labels: author: Iguazio maintainers: [] marketplaceType: '' -mlrunVersion: 1.7.2 +mlrunVersion: 1.10.0 name: onnx_utils platformVersion: 3.5.0 spec: @@ -30,8 +30,8 @@ spec: - tqdm~=4.67.1 - tensorflow~=2.19.0 - tf_keras~=2.19.0 - - torch~=2.6.0 - - torchvision~=0.21.0 + - torch~=2.8.0 + - torchvision~=0.23.0 - onnx~=1.17.0 - onnxruntime~=1.19.2 - onnxoptimizer~=0.3.13 @@ -39,4 +39,4 @@ spec: - tf2onnx~=1.16.1 - plotly~=5.23 url: '' -version: 1.3.0 +version: 1.4.0 diff --git a/functions/src/onnx_utils/onnx_utils.ipynb b/functions/src/onnx_utils/onnx_utils.ipynb index 78203a45d..14c810fab 100644 --- a/functions/src/onnx_utils/onnx_utils.ipynb +++ b/functions/src/onnx_utils/onnx_utils.ipynb @@ -77,9 +77,9 @@ "source": [ "### 1.2. Demo\n", "\n", - "We will use the `TF.Keras` framework, a `MobileNetV2` as our model and we will convert it to ONNX using the `to_onnx` handler.\n", + "We will use the `PyTorch` framework, a `MobileNetV2` as our model and we will convert it to ONNX using the `to_onnx` handler.\n", "\n", - "1.2.1. First we will set a temporary artifact path for our model to be saved in and choose the models names:" + "1.2.1. First we will set the artifact path for our model to be saved in and choose the models names:" ] }, { @@ -87,16 +87,21 @@ "metadata": { "pycharm": { "name": "#%%\n" + }, + "ExecuteTime": { + "end_time": "2026-02-10T14:13:28.256582Z", + "start_time": "2026-02-10T14:13:28.250886Z" } }, "source": [ "import os\n", - "os.environ[\"TF_USE_LEGACY_KERAS\"] = \"true\"\n", - "from tempfile import TemporaryDirectory\n", + "import tempfile\n", + "# Use a temporary directory for model artifacts (safe cleanup):\n", + "ARTIFACT_PATH = tempfile.mkdtemp()\n", + "os.environ[\"MLRUN_ARTIFACT_PATH\"] = ARTIFACT_PATH\n", "\n", - "# Create a temporary directory for the model artifact:\n", - "ARTIFACT_PATH = TemporaryDirectory().name\n", - "os.makedirs(ARTIFACT_PATH)\n", + "# Project name:\n", + "PROJECT_NAME = \"onnx-utils\"\n", "\n", "# Choose our model's name:\n", "MODEL_NAME = \"mobilenetv2\"\n", @@ -108,7 +113,7 @@ "OPTIMIZED_ONNX_MODEL_NAME = \"optimized_onnx_mobilenetv2\"" ], "outputs": [], - "execution_count": null + "execution_count": 1 }, { "cell_type": "markdown", @@ -118,87 +123,88 @@ } }, "source": [ - "1.2.2. Download the model from `keras.applications` and log it with MLRun's `TFKerasModelHandler`:" + "1.2.2. Download the model from `torchvision.models` and log it with MLRun's `PyTorchModelHandler`:" ] }, { - "cell_type": "code", "metadata": { - "pycharm": { - "name": "#%%\n" + "ExecuteTime": { + "end_time": "2026-02-10T14:00:15.032590Z", + "start_time": "2026-02-10T14:00:15.031196Z" } }, - "source": [ - "# mlrun: start-code" - ], + "cell_type": "code", + "source": "# mlrun: start-code", "outputs": [], - "execution_count": null + "execution_count": 8 }, { + "metadata": { + "ExecuteTime": { + "end_time": "2026-02-10T14:14:00.992001Z", + "start_time": "2026-02-10T14:13:33.115438Z" + } + }, "cell_type": "code", - "metadata": {}, "source": [ - "from tensorflow import keras\n", + "import torchvision\n", "\n", "import mlrun\n", - "import mlrun.frameworks.tf_keras as mlrun_tf_keras\n", + "from mlrun.frameworks.pytorch import PyTorchModelHandler\n", "\n", "\n", "def get_model(context: mlrun.MLClientCtx, model_name: str):\n", " # Download the MobileNetV2 model:\n", - " model = keras.applications.mobilenet_v2.MobileNetV2()\n", + " model = torchvision.models.mobilenet_v2()\n", "\n", " # Initialize a model handler for logging the model:\n", - " model_handler = mlrun_tf_keras.TFKerasModelHandler(\n", + " model_handler = PyTorchModelHandler(\n", " model_name=model_name,\n", " model=model,\n", - " context=context\n", + " model_class=\"mobilenet_v2\",\n", + " modules_map={\"torchvision.models\": \"mobilenet_v2\"},\n", + " context=context,\n", " )\n", "\n", " # Log the model:\n", " model_handler.log()" ], "outputs": [], - "execution_count": null + "execution_count": 2 }, { - "cell_type": "code", "metadata": { - "pycharm": { - "name": "#%%\n" + "ExecuteTime": { + "end_time": "2026-02-10T14:00:15.040221Z", + "start_time": "2026-02-10T14:00:15.038886Z" } }, - "source": [ - "# mlrun: end-code" - ], + "cell_type": "code", + "source": "# mlrun: end-code", "outputs": [], - "execution_count": null - }, - { - "cell_type": "markdown", - "metadata": { - "pycharm": { - "name": "#%% md\n" - } - }, - "source": [ - "1.2.3. Create the function using MLRun's `code_to_function` and run it:" - ] + "execution_count": 10 }, { "cell_type": "code", "metadata": { "pycharm": { "name": "#%%\n" + }, + "ExecuteTime": { + "end_time": "2026-02-10T14:14:34.429194Z", + "start_time": "2026-02-10T14:14:07.906087Z" } }, "source": [ "import mlrun\n", "\n", + "# Create or get the MLRun project:\n", + "project = mlrun.get_or_create_project(PROJECT_NAME, context=\"./\")\n", "\n", "# Create the function parsing this notebook's code using 'code_to_function':\n", "get_model_function = mlrun.code_to_function(\n", " name=\"get_mobilenetv2\",\n", + " project=PROJECT_NAME,\n", " kind=\"job\",\n", " image=\"mlrun/ml-models\"\n", ")\n", @@ -206,15 +212,267 @@ "# Run the function to log the model:\n", "get_model_run = get_model_function.run(\n", " handler=\"get_model\",\n", - " artifact_path=ARTIFACT_PATH,\n", + " output_path=ARTIFACT_PATH,\n", " params={\n", " \"model_name\": MODEL_NAME\n", " },\n", " local=True\n", ")" ], - "outputs": [], - "execution_count": null + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-10 16:14:24,932 [info] Created and saved project: {\"context\":\"./\",\"from_template\":null,\"name\":\"onnx-utils\",\"overwrite\":false,\"save\":true}\n", + "> 2026-02-10 16:14:24,933 [info] Project created successfully: {\"project_name\":\"onnx-utils\",\"stored_in_db\":true}\n", + "> 2026-02-10 16:14:31,659 [info] Storing function: {\"db\":null,\"name\":\"get-mobilenetv2-get-model\",\"uid\":\"7b9d1b54375b44e191d73685a382c910\"}\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ], + "text/html": [ + "\n", + "

\n", + "
\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
projectuiditerstartendstatekindnamelabelsinputsparametersresultsartifact_uris
onnx-utils0Feb 10 14:14:32NaTcompletedrunget-mobilenetv2-get-model
v3io_user=omerm
kind=local
owner=omerm
host=M-KCX16N69X3
model_name=mobilenetv2
mobilenetv2_modules_map.json=store://artifacts/onnx-utils/#0@7b9d1b54375b44e191d73685a382c910
model=store://models/onnx-utils/mobilenetv2#0@7b9d1b54375b44e191d73685a382c910^e0393bc5b070fd55cc57cecb94160ce412498e0f
\n", + "
\n", + "
\n", + "
\n", + " Title\n", + " ×\n", + "
\n", + " \n", + "
\n", + "
\n" + ] + }, + "metadata": {}, + "output_type": "display_data", + "jetTransient": { + "display_id": null + } + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ], + "text/html": [ + " > to track results use the .show() or .logs() methods or click here to open in UI" + ] + }, + "metadata": {}, + "output_type": "display_data", + "jetTransient": { + "display_id": null + } + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-10 16:14:34,427 [info] Run execution finished: {\"name\":\"get-mobilenetv2-get-model\",\"status\":\"completed\"}\n" + ] + } + ], + "execution_count": 3 }, { "cell_type": "markdown", @@ -228,33 +486,271 @@ "metadata": { "pycharm": { "name": "#%%\n" + }, + "ExecuteTime": { + "end_time": "2026-02-10T14:14:53.863947Z", + "start_time": "2026-02-10T14:14:48.088349Z" } }, - "source": [ - "# Import the ONNX function from the marketplace:\n", - "onnx_utils_function = mlrun.import_function(\"hub://onnx_utils\")\n", - "\n", - "# Run the function to convert our model to ONNX:\n", - "to_onnx_run = onnx_utils_function.run(\n", - " handler=\"to_onnx\",\n", - " artifact_path=ARTIFACT_PATH,\n", - " params={\n", - " \"model_name\": MODEL_NAME,\n", - " \"model_path\": get_model_run.outputs[MODEL_NAME], # <- Take the logged model from the previous function.\n", - " \"onnx_model_name\": ONNX_MODEL_NAME,\n", - " \"optimize_model\": False # <- For optimizing it later in the demo, we mark the flag as False\n", - " },\n", - " local=True\n", - ")" + "source": "# Import the ONNX function from the marketplace:\nonnx_utils_function = mlrun.import_function(\"hub://onnx_utils\", project=PROJECT_NAME)\n\n# Construct the model path from the run directory structure:\nmodel_path = os.path.join(ARTIFACT_PATH, \"get-mobilenetv2-get-model\", \"0\", \"model\")\nmodules_map_path = os.path.join(ARTIFACT_PATH, \"get-mobilenetv2-get-model\", \"0\", \"mobilenetv2_modules_map.json.json\")\n\n# Run the function to convert our model to ONNX:\nto_onnx_run = onnx_utils_function.run(\n handler=\"to_onnx\",\n output_path=ARTIFACT_PATH,\n params={\n \"model_name\": MODEL_NAME,\n \"model_path\": model_path,\n \"load_model_kwargs\": {\n \"model_name\": MODEL_NAME,\n \"model_class\": \"mobilenet_v2\",\n \"modules_map\": modules_map_path,\n },\n \"onnx_model_name\": ONNX_MODEL_NAME,\n \"optimize_model\": False, # <- For optimizing it later in the demo, we mark the flag as False\n \"framework_kwargs\": {\"input_signature\": [((32, 3, 224, 224), \"float32\")]},\n },\n local=True\n)", + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-10 16:14:48,519 [info] Storing function: {\"db\":null,\"name\":\"onnx-utils-to-onnx\",\"uid\":\"95deb2c7dbf0460291efb25c48eeebd7\"}\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ], + "text/html": [ + "\n", + "
\n", + "
\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
projectuiditerstartendstatekindnamelabelsinputsparametersresultsartifact_uris
onnx-utils0Feb 10 14:14:49NaTcompletedrunonnx-utils-to-onnx
v3io_user=omerm
kind=local
owner=omerm
host=M-KCX16N69X3
model_name=mobilenetv2
model_path=/var/folders/rn/q8gs952n26982d36y50w_2rw0000gp/T/tmpvs5qvbxr/get-mobilenetv2-get-model/0/model
load_model_kwargs={'model_name': 'mobilenetv2', 'model_class': 'mobilenet_v2', 'modules_map': '/var/folders/rn/q8gs952n26982d36y50w_2rw0000gp/T/tmpvs5qvbxr/get-mobilenetv2-get-model/0/mobilenetv2_modules_map.json.json'}
onnx_model_name=onnx_mobilenetv2
optimize_model=False
framework_kwargs={'input_signature': [((32, 3, 224, 224), 'float32')]}
model=store://models/onnx-utils/onnx_mobilenetv2#0@95deb2c7dbf0460291efb25c48eeebd7^03e4286da44d015cf5465d43e809a504d15f7f63
\n", + "
\n", + "
\n", + "
\n", + " Title\n", + " ×\n", + "
\n", + " \n", + "
\n", + "
\n" + ] + }, + "metadata": {}, + "output_type": "display_data", + "jetTransient": { + "display_id": null + } + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ], + "text/html": [ + " > to track results use the .show() or .logs() methods or click here to open in UI" + ] + }, + "metadata": {}, + "output_type": "display_data", + "jetTransient": { + "display_id": null + } + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-10 16:14:53,862 [info] Run execution finished: {\"name\":\"onnx-utils-to-onnx\",\"status\":\"completed\"}\n" + ] + } ], - "outputs": [], - "execution_count": null + "execution_count": 4 }, { "cell_type": "markdown", "metadata": {}, "source": [ - "1.2.5. Now, listing the artifact directory we will see both our `tf.keras` model and the `onnx` model:" + "1.2.5. Now we verify the ONNX model was created:" ] }, { @@ -262,16 +758,29 @@ "metadata": { "pycharm": { "name": "#%%\n" + }, + "ExecuteTime": { + "end_time": "2026-02-10T14:14:56.820411Z", + "start_time": "2026-02-10T14:14:56.817892Z" } }, "source": [ "import os\n", "\n", - "\n", - "print(os.listdir(ARTIFACT_PATH))" + "onnx_model_file = os.path.join(ARTIFACT_PATH, \"onnx-utils-to-onnx\", \"0\", \"model\", \"onnx_mobilenetv2.onnx\")\n", + "assert os.path.isfile(onnx_model_file), f\"ONNX model not found at {onnx_model_file}\"\n", + "print(f\"ONNX model created at: {onnx_model_file}\")" ], - "outputs": [], - "execution_count": null + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "ONNX model created at: /var/folders/rn/q8gs952n26982d36y50w_2rw0000gp/T/tmpvs5qvbxr/onnx-utils-to-onnx/0/model/onnx_mobilenetv2.onnx\n" + ] + } + ], + "execution_count": 5 }, { "cell_type": "markdown", @@ -308,28 +817,281 @@ "metadata": { "pycharm": { "name": "#%%\n" + }, + "ExecuteTime": { + "end_time": "2026-02-10T14:15:03.415997Z", + "start_time": "2026-02-10T14:15:00.637332Z" } }, - "source": [ - "onnx_utils_function.run(\n", - " handler=\"optimize\",\n", - " artifact_path=ARTIFACT_PATH,\n", - " params={\n", - " \"model_name\": ONNX_MODEL_NAME,\n", - " \"model_path\": to_onnx_run.output(ONNX_MODEL_NAME), # <- Take the logged model from the previous function.\n", - " \"optimized_model_name\": OPTIMIZED_ONNX_MODEL_NAME,\n", - " },\n", - " local=True\n", - ")" + "source": "# Construct the ONNX model path from the run directory structure:\nonnx_model_path = os.path.join(ARTIFACT_PATH, \"onnx-utils-to-onnx\", \"0\", \"model\")\n\nonnx_utils_function.run(\n handler=\"optimize\",\n output_path=ARTIFACT_PATH,\n params={\n \"model_path\": onnx_model_path,\n \"handler_init_kwargs\": {\"model_name\": ONNX_MODEL_NAME},\n \"optimized_model_name\": OPTIMIZED_ONNX_MODEL_NAME,\n },\n local=True\n)", + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-10 16:15:00,639 [info] Storing function: {\"db\":null,\"name\":\"onnx-utils-optimize\",\"uid\":\"0c30d7af94814dcabde8152a1951fb5d\"}\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ], + "text/html": [ + "\n", + "
\n", + "
\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
projectuiditerstartendstatekindnamelabelsinputsparametersresultsartifact_uris
onnx-utils0Feb 10 14:15:01NaTcompletedrunonnx-utils-optimize
v3io_user=omerm
kind=local
owner=omerm
host=M-KCX16N69X3
model_path=/var/folders/rn/q8gs952n26982d36y50w_2rw0000gp/T/tmpvs5qvbxr/onnx-utils-to-onnx/0/model
handler_init_kwargs={'model_name': 'onnx_mobilenetv2'}
optimized_model_name=optimized_onnx_mobilenetv2
model=store://models/onnx-utils/optimized_onnx_mobilenetv2#0@0c30d7af94814dcabde8152a1951fb5d^599547984e83a664dc1c2708607d06731edb5ac2
\n", + "
\n", + "
\n", + "
\n", + " Title\n", + " ×\n", + "
\n", + " \n", + "
\n", + "
\n" + ] + }, + "metadata": {}, + "output_type": "display_data", + "jetTransient": { + "display_id": null + } + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ], + "text/html": [ + " > to track results use the .show() or .logs() methods or click here to open in UI" + ] + }, + "metadata": {}, + "output_type": "display_data", + "jetTransient": { + "display_id": null + } + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "> 2026-02-10 16:15:03,414 [info] Run execution finished: {\"name\":\"onnx-utils-optimize\",\"status\":\"completed\"}\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ] + }, + "execution_count": 6, + "metadata": {}, + "output_type": "execute_result" + } ], - "outputs": [], - "execution_count": null + "execution_count": 6 }, { "cell_type": "markdown", "metadata": {}, "source": [ - "2.2.2. And now our model was optimized and can be seen under the `ARTIFACT_PATH`:" + "2.2.2. And now our model was optimized. Let us verify:" ] }, { @@ -337,13 +1099,27 @@ "metadata": { "pycharm": { "name": "#%%\n" + }, + "ExecuteTime": { + "end_time": "2026-02-10T14:15:05.748413Z", + "start_time": "2026-02-10T14:15:05.745309Z" } }, "source": [ - "print(os.listdir(ARTIFACT_PATH))" + "optimized_model_file = os.path.join(ARTIFACT_PATH, \"onnx-utils-optimize\", \"0\", \"model\", \"optimized_onnx_mobilenetv2.onnx\")\n", + "assert os.path.isfile(optimized_model_file), f\"Optimized ONNX model not found at {optimized_model_file}\"\n", + "print(f\"Optimized ONNX model created at: {optimized_model_file}\")" ], - "outputs": [], - "execution_count": null + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Optimized ONNX model created at: /var/folders/rn/q8gs952n26982d36y50w_2rw0000gp/T/tmpvs5qvbxr/onnx-utils-optimize/0/model/optimized_onnx_mobilenetv2.onnx\n" + ] + } + ], + "execution_count": 7 }, { "cell_type": "markdown", @@ -353,7 +1129,7 @@ } }, "source": [ - "Lastly, run this code to clean up the models:" + "Lastly, run this code to clean up all generated files and directories:" ] }, { @@ -361,23 +1137,22 @@ "metadata": { "pycharm": { "name": "#%%\n" + }, + "ExecuteTime": { + "end_time": "2026-02-10T14:00:28.409998Z", + "start_time": "2026-02-10T13:57:21.679146Z" } }, - "source": [ - "import shutil\n", - "\n", - "\n", - "shutil.rmtree(ARTIFACT_PATH)" - ], + "source": "import shutil\n\n# Clean up the temporary artifact directory:\nif os.path.exists(ARTIFACT_PATH):\n shutil.rmtree(ARTIFACT_PATH)", "outputs": [], "execution_count": null } ], "metadata": { "kernelspec": { - "display_name": "Python 3 (ipykernel)", + "display_name": "mlrun_functions", "language": "python", - "name": "python3" + "name": "mlrun_functions" }, "language_info": { "codemirror_mode": { diff --git a/functions/src/onnx_utils/requirements.txt b/functions/src/onnx_utils/requirements.txt index d3d7dfd68..912b3d7e5 100644 --- a/functions/src/onnx_utils/requirements.txt +++ b/functions/src/onnx_utils/requirements.txt @@ -1,11 +1,10 @@ tqdm~=4.67.1 tensorflow~=2.19.0 tf_keras~=2.19.0 -torch~=2.6.0 -torchvision~=0.21.0 +torch~=2.8 +torchvision~=0.23.0 onnx~=1.17.0 onnxruntime~=1.19.2 onnxoptimizer~=0.3.13 onnxmltools~=1.13.0 -tf2onnx~=1.16.1 -plotly~=5.23 +plotly~=5.23 \ No newline at end of file diff --git a/functions/src/onnx_utils/test_onnx_utils.py b/functions/src/onnx_utils/test_onnx_utils.py index 2e01782f5..59c6c2b38 100644 --- a/functions/src/onnx_utils/test_onnx_utils.py +++ b/functions/src/onnx_utils/test_onnx_utils.py @@ -17,6 +17,9 @@ import tempfile import mlrun +import pytest + +PROJECT_NAME = "onnx-utils" # Choose our model's name: MODEL_NAME = "model" @@ -27,41 +30,67 @@ # Choose our optimized ONNX version model's name: OPTIMIZED_ONNX_MODEL_NAME = f"optimized_{ONNX_MODEL_NAME}" +REQUIRED_ENV_VARS = [ + "MLRUN_DBPATH", + "MLRUN_ARTIFACT_PATH", + "V3IO_USERNAME", + "V3IO_ACCESS_KEY", +] -def _setup_environment() -> str: - """ - Setup the test environment, creating the artifacts path of the test. - :returns: The temporary directory created for the test artifacts path. +def _validate_environment_variables() -> bool: """ - artifact_path = tempfile.TemporaryDirectory().name - os.makedirs(artifact_path) - return artifact_path + Checks that all required Environment variables are set. + """ + environment_keys = os.environ.keys() + return all(key in environment_keys for key in REQUIRED_ENV_VARS) -def _cleanup_environment(artifact_path: str): +def _is_tf2onnx_available() -> bool: """ - Cleanup the test environment, deleting files and artifacts created during the test. - - :param artifact_path: The artifact path to delete. + Check if tf2onnx is installed (required for TensorFlow/Keras ONNX conversion). """ - # Clean the local directory: + try: + import tf2onnx + return True + except ImportError: + return False + + +@pytest.fixture(scope="session") +def onnx_project(): + """Create/get the MLRun project once per test session.""" + return mlrun.get_or_create_project(PROJECT_NAME, context="./") + + +@pytest.fixture(autouse=True) +def test_environment(onnx_project): + """Setup and cleanup test artifacts for each test.""" + artifact_path = tempfile.mkdtemp() + yield artifact_path + # Cleanup - only remove files/dirs from the directory containing this test file, + # never from an arbitrary CWD (which could be the project root). + test_dir = os.path.dirname(os.path.abspath(__file__)) for test_output in [ - *os.listdir(artifact_path), "schedules", "runs", "artifacts", "functions", + "model.pt", + "model.zip", + "model_modules_map.json", + "model_modules_map.json.json", + "onnx_model.onnx", + "optimized_onnx_model.onnx", ]: - test_output_path = os.path.abspath(f"./{test_output}") + test_output_path = os.path.join(test_dir, test_output) if os.path.exists(test_output_path): if os.path.isdir(test_output_path): shutil.rmtree(test_output_path) else: os.remove(test_output_path) - - # Clean the artifacts directory: - shutil.rmtree(artifact_path) + if os.path.exists(artifact_path): + shutil.rmtree(artifact_path) def _log_tf_keras_model(context: mlrun.MLClientCtx, model_name: str): @@ -114,42 +143,55 @@ def _log_pytorch_model(context: mlrun.MLClientCtx, model_name: str): model_handler.log() -def test_to_onnx_help(): +@pytest.mark.skipif( + condition=not _validate_environment_variables(), + reason="Project's environment variables are not set", +) +def test_to_onnx_help(test_environment): """ Test the 'to_onnx' handler, passing "help" in the 'framework_kwargs'. """ - # Setup the tests environment: - artifact_path = _setup_environment() + artifact_path = test_environment # Create the function: log_model_function = mlrun.code_to_function( filename="test_onnx_utils.py", name="log_model", + project=PROJECT_NAME, kind="job", image="mlrun/ml-models", ) # Run the function to log the model: - log_model_run = log_model_function.run( - handler="_log_tf_keras_model", - artifact_path=artifact_path, + log_model_function.run( + handler="_log_pytorch_model", + output_path=artifact_path, params={"model_name": MODEL_NAME}, local=True, ) + # Get artifact paths - construct from artifact_path and run structure + run_artifact_dir = os.path.join(artifact_path, "log-model--log-pytorch-model", "0") + model_path = os.path.join(run_artifact_dir, "model") + modules_map_path = os.path.join(run_artifact_dir, "model_modules_map.json.json") + # Import the ONNX Utils function: - onnx_function = mlrun.import_function("function.yaml") + onnx_function = mlrun.import_function("function.yaml", project=PROJECT_NAME) # Run the function, passing "help" in 'framework_kwargs' and see that no exception was raised: is_test_passed = True try: onnx_function.run( handler="to_onnx", - artifact_path=artifact_path, + output_path=artifact_path, params={ # Take the logged model from the previous function. - "model_path": log_model_run.status.artifacts[0]["spec"]["target_path"], - "load_model_kwargs": {"model_name": MODEL_NAME}, + "model_path": model_path, + "load_model_kwargs": { + "model_name": MODEL_NAME, + "model_class": "mobilenet_v2", + "modules_map": modules_map_path, + }, "framework_kwargs": "help", }, local=True, @@ -160,23 +202,28 @@ def test_to_onnx_help(): ) is_test_passed = False - # Cleanup the tests environment: - _cleanup_environment(artifact_path=artifact_path) - assert is_test_passed -def test_tf_keras_to_onnx(): +@pytest.mark.skipif( + condition=not _validate_environment_variables(), + reason="Project's environment variables are not set", +) +@pytest.mark.skipif( + condition=not _is_tf2onnx_available(), + reason="tf2onnx is not installed", +) +def test_tf_keras_to_onnx(test_environment): """ Test the 'to_onnx' handler, giving it a tf.keras model. """ - # Setup the tests environment: - artifact_path = _setup_environment() + artifact_path = test_environment # Create the function: log_model_function = mlrun.code_to_function( filename="test_onnx_utils.py", name="log_model", + project=PROJECT_NAME, kind="job", image="mlrun/ml-models", ) @@ -184,18 +231,18 @@ def test_tf_keras_to_onnx(): # Run the function to log the model: log_model_run = log_model_function.run( handler="_log_tf_keras_model", - artifact_path=artifact_path, + output_path=artifact_path, params={"model_name": MODEL_NAME}, local=True, ) # Import the ONNX Utils function: - onnx_function = mlrun.import_function("function.yaml") + onnx_function = mlrun.import_function("function.yaml", project=PROJECT_NAME) # Run the function to convert our model to ONNX: onnx_function_run = onnx_function.run( handler="to_onnx", - artifact_path=artifact_path, + output_path=artifact_path, params={ # Take the logged model from the previous function. "model_path": log_model_run.status.artifacts[0]["spec"]["target_path"], @@ -205,9 +252,6 @@ def test_tf_keras_to_onnx(): local=True, ) - # Cleanup the tests environment: - _cleanup_environment(artifact_path=artifact_path) - # Print the outputs list: print(f"Produced outputs: {onnx_function_run.outputs}") @@ -215,17 +259,21 @@ def test_tf_keras_to_onnx(): assert "model" in onnx_function_run.outputs -def test_pytorch_to_onnx(): +@pytest.mark.skipif( + condition=not _validate_environment_variables(), + reason="Project's environment variables are not set", +) +def test_pytorch_to_onnx(test_environment): """ Test the 'to_onnx' handler, giving it a pytorch model. """ - # Setup the tests environment: - artifact_path = _setup_environment() + artifact_path = test_environment # Create the function: log_model_function = mlrun.code_to_function( filename="test_onnx_utils.py", name="log_model", + project=PROJECT_NAME, kind="job", image="mlrun/ml-models", ) @@ -233,25 +281,30 @@ def test_pytorch_to_onnx(): # Run the function to log the model: log_model_run = log_model_function.run( handler="_log_pytorch_model", - artifact_path=artifact_path, + output_path=artifact_path, params={"model_name": MODEL_NAME}, local=True, ) # Import the ONNX Utils function: - onnx_function = mlrun.import_function("function.yaml") + onnx_function = mlrun.import_function("function.yaml", project=PROJECT_NAME) + + # Get artifact paths - construct from artifact_path and run structure + run_artifact_dir = os.path.join(artifact_path, "log-model--log-pytorch-model", "0") + model_path = os.path.join(run_artifact_dir, "model") + modules_map_path = os.path.join(run_artifact_dir, "model_modules_map.json.json") # Run the function to convert our model to ONNX: onnx_function_run = onnx_function.run( handler="to_onnx", - artifact_path=artifact_path, + output_path=artifact_path, params={ # Take the logged model from the previous function. - "model_path": log_model_run.status.artifacts[1]["spec"]["target_path"], + "model_path": model_path, "load_model_kwargs": { "model_name": MODEL_NAME, "model_class": "mobilenet_v2", - "modules_map": log_model_run.status.artifacts[0]["spec"]["target_path"], + "modules_map": modules_map_path, }, "onnx_model_name": ONNX_MODEL_NAME, "framework_kwargs": {"input_signature": [((32, 3, 224, 224), "float32")]}, @@ -259,9 +312,6 @@ def test_pytorch_to_onnx(): local=True, ) - # Cleanup the tests environment: - _cleanup_environment(artifact_path=artifact_path) - # Print the outputs list: print(f"Produced outputs: {onnx_function_run.outputs}") @@ -269,22 +319,25 @@ def test_pytorch_to_onnx(): assert "model" in onnx_function_run.outputs -def test_optimize_help(): +@pytest.mark.skipif( + condition=not _validate_environment_variables(), + reason="Project's environment variables are not set", +) +def test_optimize_help(test_environment): """ Test the 'optimize' handler, passing "help" in the 'optimizations'. """ - # Setup the tests environment: - artifact_path = _setup_environment() + artifact_path = test_environment # Import the ONNX Utils function: - onnx_function = mlrun.import_function("function.yaml") + onnx_function = mlrun.import_function("function.yaml", project=PROJECT_NAME) # Run the function, passing "help" in 'optimizations' and see that no exception was raised: is_test_passed = True try: onnx_function.run( handler="optimize", - artifact_path=artifact_path, + output_path=artifact_path, params={ "model_path": "", "optimizations": "help", @@ -297,69 +350,81 @@ def test_optimize_help(): ) is_test_passed = False - # Cleanup the tests environment: - _cleanup_environment(artifact_path=artifact_path) - assert is_test_passed -def test_optimize(): +@pytest.mark.skipif( + condition=not _validate_environment_variables(), + reason="Project's environment variables are not set", +) +def test_optimize(test_environment): """ - Test the 'optimize' handler, giving it a model from the ONNX zoo git repository. + Test the 'optimize' handler, giving it a pytorch model converted to ONNX. """ - # Setup the tests environment: - artifact_path = _setup_environment() + artifact_path = test_environment # Create the function: log_model_function = mlrun.code_to_function( filename="test_onnx_utils.py", name="log_model", + project=PROJECT_NAME, kind="job", image="mlrun/ml-models", ) # Run the function to log the model: - log_model_run = log_model_function.run( - handler="_log_tf_keras_model", - artifact_path=artifact_path, + log_model_function.run( + handler="_log_pytorch_model", + output_path=artifact_path, params={"model_name": MODEL_NAME}, local=True, ) + # Get artifact paths - construct from artifact_path and run structure + run_artifact_dir = os.path.join(artifact_path, "log-model--log-pytorch-model", "0") + model_path = os.path.join(run_artifact_dir, "model") + modules_map_path = os.path.join(run_artifact_dir, "model_modules_map.json.json") + # Import the ONNX Utils function: - onnx_function = mlrun.import_function("function.yaml") + onnx_function = mlrun.import_function("function.yaml", project=PROJECT_NAME) # Run the function to convert our model to ONNX: - to_onnx_function_run = onnx_function.run( + onnx_function.run( handler="to_onnx", - artifact_path=artifact_path, + output_path=artifact_path, params={ # Take the logged model from the previous function. - "model_path": log_model_run.status.artifacts[0]["spec"]["target_path"], - "load_model_kwargs": {"model_name": MODEL_NAME}, + "model_path": model_path, + "load_model_kwargs": { + "model_name": MODEL_NAME, + "model_class": "mobilenet_v2", + "modules_map": modules_map_path, + }, "onnx_model_name": ONNX_MODEL_NAME, + "framework_kwargs": {"input_signature": [((32, 3, 224, 224), "float32")]}, }, local=True, ) + # Get the ONNX model path from the to_onnx run output + onnx_run_artifact_dir = os.path.join( + artifact_path, "onnx-utils-to-onnx", "0" + ) + onnx_model_path = os.path.join(onnx_run_artifact_dir, "model") + # Run the function to optimize our model: optimize_function_run = onnx_function.run( handler="optimize", - artifact_path=artifact_path, + output_path=artifact_path, params={ # Take the logged model from the previous function. - "model_path": to_onnx_function_run.status.artifacts[0]["spec"][ - "target_path" - ], + "model_path": onnx_model_path, "handler_init_kwargs": {"model_name": ONNX_MODEL_NAME}, "optimized_model_name": OPTIMIZED_ONNX_MODEL_NAME, }, local=True, ) - # Cleanup the tests environment: - _cleanup_environment(artifact_path=artifact_path) - # Print the outputs list: print(f"Produced outputs: {optimize_function_run.outputs}")