2024-11-15 09:43:15 -05:00
|
|
|
# Copyright (c) Microsoft. All rights reserved.
|
|
|
|
|
|
2026-02-01 17:18:19 -08:00
|
|
|
import importlib
|
|
|
|
|
import inspect
|
|
|
|
|
from collections.abc import Sequence
|
2024-11-15 09:43:15 -05:00
|
|
|
from typing import Any
|
|
|
|
|
|
2026-02-01 17:18:19 -08:00
|
|
|
from semantic_kernel.exceptions.process_exceptions import ProcessInvalidConfigurationException
|
2024-11-15 09:43:15 -05:00
|
|
|
from semantic_kernel.functions.kernel_function import KernelFunction
|
|
|
|
|
from semantic_kernel.processes.kernel_process.kernel_process_message_channel import KernelProcessMessageChannel
|
2026-02-01 17:18:19 -08:00
|
|
|
from semantic_kernel.processes.kernel_process.kernel_process_step import KernelProcessStep
|
2024-11-15 09:43:15 -05:00
|
|
|
from semantic_kernel.processes.kernel_process.kernel_process_step_context import KernelProcessStepContext
|
2025-04-29 07:51:11 +09:00
|
|
|
from semantic_kernel.utils.feature_stage_decorator import experimental
|
2024-11-15 09:43:15 -05:00
|
|
|
|
Python: Default Dapr module allowlist to semantic_kernel prefix (#13596)
### Motivation and Context
Follow-up to #13499. The previous PR added the `allowed_module_prefixes`
parameter but defaulted it to `None`, which meant the module restriction
was only active if developers discovered and configured it.
Secure-by-default is the right posture here — restrict first, let
developers widen as needed.
- Change `allowed_module_prefixes` default from `None` to
`("semantic_kernel.",)` across Dapr runtime step loading
- Non-SK step classes now require developers to explicitly add their
module prefix (e.g. `("semantic_kernel.", "myapp.steps.")`)
- Developers can pass `None` to opt out entirely, but the secure default
is now enforced
- The Dapr runtime code is experimental, so this is a non-breaking
change per our stability guarantees
<!-- Thank you for your contribution to the semantic-kernel repo!
Please help reviewers and future users, providing the following
information:
1. Why is this change required?
2. What problem does it solve?
3. What scenario does it contribute to?
4. If it fixes an open issue, please link to the issue here.
-->
<!-- Describe your changes, the overall approach, the underlying design.
These notes will help understanding how your code works. Thanks! -->
### Contribution Checklist
<!-- Before submitting this PR, please make sure: -->
- [X] The code builds clean without any errors or warnings
- [X] The PR follows the [SK Contribution
Guidelines](https://github.com/microsoft/semantic-kernel/blob/main/CONTRIBUTING.md)
and the [pre-submission formatting
script](https://github.com/microsoft/semantic-kernel/blob/main/CONTRIBUTING.md#development-scripts)
raises no violations
- [X] All unit tests pass, and I have added new tests where possible
- [ ] I didn't break anyone :smile:
---------
Co-authored-by: MAF Dashboard Bot <maf-dashboard-bot@users.noreply.github.com>
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-03-24 11:34:25 +09:00
|
|
|
DEFAULT_ALLOWED_MODULE_PREFIXES: tuple[str, ...] = ("semantic_kernel.",)
|
|
|
|
|
|
2024-11-15 09:43:15 -05:00
|
|
|
|
2025-04-29 07:51:11 +09:00
|
|
|
@experimental
|
2024-11-15 09:43:15 -05:00
|
|
|
def find_input_channels(
|
|
|
|
|
channel: KernelProcessMessageChannel, functions: dict[str, KernelFunction]
|
|
|
|
|
) -> dict[str, dict[str, Any | None]]:
|
|
|
|
|
"""Finds and creates input channels."""
|
|
|
|
|
if not functions:
|
|
|
|
|
raise ValueError("The step has not been initialized.")
|
|
|
|
|
|
|
|
|
|
inputs: dict[str, Any] = {}
|
|
|
|
|
for name, function in functions.items():
|
|
|
|
|
inputs[name] = {}
|
|
|
|
|
for param in function.metadata.parameters:
|
|
|
|
|
# Check for Kernel, and skip if necessary, since it is populated later on
|
|
|
|
|
if param.type_ == "Kernel":
|
|
|
|
|
continue
|
|
|
|
|
if not param.is_required:
|
|
|
|
|
continue
|
|
|
|
|
if param.type_ == "KernelProcessStepContext":
|
|
|
|
|
inputs[name][param.name] = KernelProcessStepContext(channel)
|
|
|
|
|
else:
|
|
|
|
|
inputs[name][param.name] = None
|
|
|
|
|
|
|
|
|
|
return inputs
|
|
|
|
|
|
|
|
|
|
|
2025-04-29 07:51:11 +09:00
|
|
|
@experimental
|
2024-11-15 09:43:15 -05:00
|
|
|
def get_fully_qualified_name(cls) -> str:
|
|
|
|
|
"""Gets the fully qualified name of a class."""
|
|
|
|
|
return f"{cls.__module__}.{cls.__name__}"
|
2026-02-01 17:18:19 -08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@experimental
|
|
|
|
|
def get_step_class_from_qualified_name(
|
|
|
|
|
full_class_name: str,
|
Python: Default Dapr module allowlist to semantic_kernel prefix (#13596)
### Motivation and Context
Follow-up to #13499. The previous PR added the `allowed_module_prefixes`
parameter but defaulted it to `None`, which meant the module restriction
was only active if developers discovered and configured it.
Secure-by-default is the right posture here — restrict first, let
developers widen as needed.
- Change `allowed_module_prefixes` default from `None` to
`("semantic_kernel.",)` across Dapr runtime step loading
- Non-SK step classes now require developers to explicitly add their
module prefix (e.g. `("semantic_kernel.", "myapp.steps.")`)
- Developers can pass `None` to opt out entirely, but the secure default
is now enforced
- The Dapr runtime code is experimental, so this is a non-breaking
change per our stability guarantees
<!-- Thank you for your contribution to the semantic-kernel repo!
Please help reviewers and future users, providing the following
information:
1. Why is this change required?
2. What problem does it solve?
3. What scenario does it contribute to?
4. If it fixes an open issue, please link to the issue here.
-->
<!-- Describe your changes, the overall approach, the underlying design.
These notes will help understanding how your code works. Thanks! -->
### Contribution Checklist
<!-- Before submitting this PR, please make sure: -->
- [X] The code builds clean without any errors or warnings
- [X] The PR follows the [SK Contribution
Guidelines](https://github.com/microsoft/semantic-kernel/blob/main/CONTRIBUTING.md)
and the [pre-submission formatting
script](https://github.com/microsoft/semantic-kernel/blob/main/CONTRIBUTING.md#development-scripts)
raises no violations
- [X] All unit tests pass, and I have added new tests where possible
- [ ] I didn't break anyone :smile:
---------
Co-authored-by: MAF Dashboard Bot <maf-dashboard-bot@users.noreply.github.com>
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-03-24 11:34:25 +09:00
|
|
|
allowed_module_prefixes: Sequence[str] | None = DEFAULT_ALLOWED_MODULE_PREFIXES,
|
2026-02-01 17:18:19 -08:00
|
|
|
) -> type[KernelProcessStep]:
|
|
|
|
|
"""Loads and validates a KernelProcessStep class from a fully qualified name.
|
|
|
|
|
|
|
|
|
|
This function validates that the loaded class is a proper subclass of
|
|
|
|
|
KernelProcessStep, preventing instantiation of arbitrary classes.
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
full_class_name: The fully qualified class name in Python import notation
|
|
|
|
|
(e.g., 'mypackage.mymodule.MyStep'). The module must be importable
|
|
|
|
|
from the current Python environment.
|
Python: Default Dapr module allowlist to semantic_kernel prefix (#13596)
### Motivation and Context
Follow-up to #13499. The previous PR added the `allowed_module_prefixes`
parameter but defaulted it to `None`, which meant the module restriction
was only active if developers discovered and configured it.
Secure-by-default is the right posture here — restrict first, let
developers widen as needed.
- Change `allowed_module_prefixes` default from `None` to
`("semantic_kernel.",)` across Dapr runtime step loading
- Non-SK step classes now require developers to explicitly add their
module prefix (e.g. `("semantic_kernel.", "myapp.steps.")`)
- Developers can pass `None` to opt out entirely, but the secure default
is now enforced
- The Dapr runtime code is experimental, so this is a non-breaking
change per our stability guarantees
<!-- Thank you for your contribution to the semantic-kernel repo!
Please help reviewers and future users, providing the following
information:
1. Why is this change required?
2. What problem does it solve?
3. What scenario does it contribute to?
4. If it fixes an open issue, please link to the issue here.
-->
<!-- Describe your changes, the overall approach, the underlying design.
These notes will help understanding how your code works. Thanks! -->
### Contribution Checklist
<!-- Before submitting this PR, please make sure: -->
- [X] The code builds clean without any errors or warnings
- [X] The PR follows the [SK Contribution
Guidelines](https://github.com/microsoft/semantic-kernel/blob/main/CONTRIBUTING.md)
and the [pre-submission formatting
script](https://github.com/microsoft/semantic-kernel/blob/main/CONTRIBUTING.md#development-scripts)
raises no violations
- [X] All unit tests pass, and I have added new tests where possible
- [ ] I didn't break anyone :smile:
---------
Co-authored-by: MAF Dashboard Bot <maf-dashboard-bot@users.noreply.github.com>
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-03-24 11:34:25 +09:00
|
|
|
allowed_module_prefixes: Sequence of module prefixes that are allowed
|
|
|
|
|
to be imported. The module must start with one of these prefixes.
|
|
|
|
|
This check is performed BEFORE import to prevent execution of
|
|
|
|
|
module-level code in unauthorized modules. Defaults to
|
|
|
|
|
("semantic_kernel.",). Pass None to allow any module (not
|
|
|
|
|
recommended for production). An empty sequence blocks all modules.
|
2026-02-01 17:18:19 -08:00
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
The validated class type that is a subclass of KernelProcessStep
|
|
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
|
ProcessInvalidConfigurationException: Raised when:
|
|
|
|
|
- The class name format is invalid (missing module separator)
|
|
|
|
|
- The module is not in the allowed prefixes list (if provided)
|
|
|
|
|
- The module cannot be imported
|
|
|
|
|
- The class attribute doesn't exist in the module
|
|
|
|
|
- The attribute is not a class type
|
|
|
|
|
- The class is not a subclass of KernelProcessStep
|
|
|
|
|
"""
|
|
|
|
|
if not full_class_name or "." not in full_class_name:
|
|
|
|
|
raise ProcessInvalidConfigurationException(
|
|
|
|
|
f"Invalid step class name format: '{full_class_name}'. "
|
|
|
|
|
"Expected a fully qualified name like 'module.ClassName'."
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
module_name, class_name = full_class_name.rsplit(".", 1)
|
|
|
|
|
|
|
|
|
|
if not module_name or not class_name:
|
|
|
|
|
raise ProcessInvalidConfigurationException(
|
|
|
|
|
f"Invalid step class name format: '{full_class_name}'. Module name and class name cannot be empty."
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Check module allowlist BEFORE import to prevent module-level code execution
|
Python: Default Dapr module allowlist to semantic_kernel prefix (#13596)
### Motivation and Context
Follow-up to #13499. The previous PR added the `allowed_module_prefixes`
parameter but defaulted it to `None`, which meant the module restriction
was only active if developers discovered and configured it.
Secure-by-default is the right posture here — restrict first, let
developers widen as needed.
- Change `allowed_module_prefixes` default from `None` to
`("semantic_kernel.",)` across Dapr runtime step loading
- Non-SK step classes now require developers to explicitly add their
module prefix (e.g. `("semantic_kernel.", "myapp.steps.")`)
- Developers can pass `None` to opt out entirely, but the secure default
is now enforced
- The Dapr runtime code is experimental, so this is a non-breaking
change per our stability guarantees
<!-- Thank you for your contribution to the semantic-kernel repo!
Please help reviewers and future users, providing the following
information:
1. Why is this change required?
2. What problem does it solve?
3. What scenario does it contribute to?
4. If it fixes an open issue, please link to the issue here.
-->
<!-- Describe your changes, the overall approach, the underlying design.
These notes will help understanding how your code works. Thanks! -->
### Contribution Checklist
<!-- Before submitting this PR, please make sure: -->
- [X] The code builds clean without any errors or warnings
- [X] The PR follows the [SK Contribution
Guidelines](https://github.com/microsoft/semantic-kernel/blob/main/CONTRIBUTING.md)
and the [pre-submission formatting
script](https://github.com/microsoft/semantic-kernel/blob/main/CONTRIBUTING.md#development-scripts)
raises no violations
- [X] All unit tests pass, and I have added new tests where possible
- [ ] I didn't break anyone :smile:
---------
Co-authored-by: MAF Dashboard Bot <maf-dashboard-bot@users.noreply.github.com>
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-03-24 11:34:25 +09:00
|
|
|
if allowed_module_prefixes is not None and not any(
|
|
|
|
|
module_name.startswith(prefix)
|
|
|
|
|
if prefix.endswith(".")
|
|
|
|
|
else (module_name == prefix or module_name.startswith(prefix + "."))
|
|
|
|
|
for prefix in allowed_module_prefixes
|
|
|
|
|
):
|
2026-02-01 17:18:19 -08:00
|
|
|
raise ProcessInvalidConfigurationException(
|
|
|
|
|
f"Module '{module_name}' is not in the allowed module prefixes: {allowed_module_prefixes}. "
|
|
|
|
|
f"Step class '{full_class_name}' cannot be loaded."
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
module = importlib.import_module(module_name)
|
|
|
|
|
except ImportError as e:
|
|
|
|
|
raise ProcessInvalidConfigurationException(
|
|
|
|
|
f"Unable to import module '{module_name}' for step class '{full_class_name}': {e}"
|
|
|
|
|
) from e
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
cls = getattr(module, class_name)
|
|
|
|
|
except AttributeError as e:
|
|
|
|
|
raise ProcessInvalidConfigurationException(
|
|
|
|
|
f"Class '{class_name}' not found in module '{module_name}': {e}"
|
|
|
|
|
) from e
|
|
|
|
|
|
|
|
|
|
if not inspect.isclass(cls):
|
|
|
|
|
raise ProcessInvalidConfigurationException(f"'{full_class_name}' is not a class type, got {type(cls).__name__}")
|
|
|
|
|
|
|
|
|
|
if not issubclass(cls, KernelProcessStep):
|
|
|
|
|
raise ProcessInvalidConfigurationException(
|
|
|
|
|
f"Step class '{full_class_name}' must be a subclass of KernelProcessStep. Got: {cls.__bases__}"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
return cls
|