maybenot


Namemaybenot JSON
Version 0.5.0rc1 PyPI version JSON
download
home_pagehttps://github.com/histrio/maybenot
SummarySee what a program does before deciding whether you really want it to happen.
upload_time2024-04-10 08:23:38
maintainerNone
docs_urlNone
authorPhilipp Emanuel Weidmann
requires_python<4.0,>=3.8
licenseGPLv3
keywords command preview system calls interception file system operations simulation dry run utility
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            [![Release](https://img.shields.io/github/v/release/histrio/maybenot)](https://img.shields.io/github/v/release/histrio/maybenot)
[![Tests](https://github.com/histrio/maybenot/actions/workflows/main.yml/badge.svg?branch=master)](https://github.com/histrio/maybenot/actions/workflows/main.yml)
[![codecov](https://codecov.io/gh/histrio/maybenot/graph/badge.svg?token=X6ZMMZNN5T)](https://codecov.io/gh/histrio/maybenot)
[![Commit activity](https://img.shields.io/github/commit-activity/m/histrio/maybenot)](https://img.shields.io/github/commit-activity/m/histrio/maybenot)
[![License](https://img.shields.io/github/license/histrio/maybenot)](https://img.shields.io/github/license/histrio/maybenot)


---


```
rm -rf pic*
```

Are you sure? Are you *one hundred percent* sure?


# `maybenot`...

... allows you to run a command and see what it does to your files *without actually doing it!* After reviewing the operations listed, you can then decide whether you really want these things to happen or not.

![Screenshot](screenshot.png)


## What is this sorcery?!?

`maybenot` runs processes under the control of [ptrace](https://en.wikipedia.org/wiki/Ptrace) (with the help of the excellent [python-ptrace](https://github.com/haypo/python-ptrace) library). When it intercepts a system call that is about to make changes to the file system, it logs that call, and then modifies CPU registers to both redirect the call to an invalid syscall ID (effectively turning it into a no-op) and set the return value of that no-op call to one indicating success of the original call.

As a result, the process believes that everything it is trying to do is actually happening, when in reality nothing is.

That being said, `maybenot` **should :warning: NEVER :warning: be used to run untrusted code** on a system you care about! A process running under `maybenot` can still do serious damage to your system because only a handful of syscalls are blocked. It can also check whether an operation such as deleting a file succeeded using read-only syscalls, and *alter its behavior accordingly.* Therefore, a rerun without restrictions is not guaranteed to always produce the displayed operations.

Currently, `maybenot` is best thought of as an (alpha-quality) "what exactly will this command I typed myself do?" tool.


## Installation

`maybenot` runs on Linux :penguin: and requires [Python 3.8+](https://www.python.org/) :snake:. If you have the [pip](https://pip.pypa.io) package manager, all you need to do is run

```
pip install maybe
```

either as a superuser or from a [virtualenv](https://virtualenv.pypa.io) environment. To develop `maybenot`, clone the repository and run

```
pip install -e .
```

in its main directory to install the package in editable mode.


## Usage

```
maybenot [options] command [argument ...]
```

### Positional arguments

| Argument | Description |
| --- | --- |
| `command` | the command to run under `maybenot`'s control |
| `argument ...` | argument(s) to pass to `command` |

### Optional arguments

| Argument | Description |
| --- | --- |
| `-a OPERATION ...`,<br>`--allow OPERATION ...`<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | allow the command to perform the specified operation(s). all other operations will be denied. possible values for `OPERATION` are: `change_owner`, `change_permissions`, `create_directory`, `create_link`, `create_write_file`, `delete`, `move`; as well as any filter scopes defined by loaded plugins |
| `-d OPERATION ...`,<br>`--deny OPERATION ...` | deny the command the specified operation(s). all other operations will be allowed. see `--allow` for a list of possible values for `OPERATION`. `--allow` and `--deny` cannot be combined |
| `-p FILE ...`,<br>`--plugin FILE ...` | load the specified [plugin](#plugin-api) script(s) |
| `-l`, `--list-only` | list operations without header, indentation and rerun prompt |
| `--style-output {yes,no,auto}` | colorize output using ANSI escape sequences (`yes`/`no`) or automatically decide based on whether stdout is a terminal (`auto`, default) |
| `-v`, `--verbose` | if specified once, print every filtered syscall. if specified twice, print every syscall, highlighting filtered syscalls |
| `--version` | show program's version number and exit |
| `-h`, `--help` | show a help message and exit |


## Plugin API

By default, `maybenot` intercepts and blocks all syscalls that can make permanent modifications to the system. For more specialized syscall filtering needs, `maybenot` provides a simple yet powerful plugin API. Filter plugins are written in pure Python and use the same interfaces as [`maybenot`'s built-in filters](maybenot/filters).

The public API is composed of the following two members:

#### *maybenot.*`T`

A [Blessings](https://github.com/erikrose/blessings) `Terminal` object that can be used to format console output (such as `operation` as documented below). Output formatted with this object automatically complies with the `--style-output` command line argument.

#### *maybenot.*`register_filter(syscall, filter_function, filter_scope=None)`

Add the filter `filter_function` to the filter registry. If the filter is enabled (which is the default, but can be altered with the `--allow` and `--deny` command line arguments), it intercepts all calls to `syscall` made by the controlled process. `filter_scope` determines the key to be used in conjunction with `--allow` and `--deny` to enable/disable the filter (multiple filters can share the same key). If `filter_scope` is omitted or `None`, the last part of the plugin's module name is used.

`filter_function` itself must conform to the signature `filter_function(process, args)`. `process` is a [`Process`](maybenot/process.py) control object that can be used to inspect and manipulate the process, while `args` is the list of arguments passed to the syscall in the order in which they appear in the syscall's signature. If an argument represents a (pointer to a) filename, the argument will be of type `str` and contain the filename, otherwise it will be of type `int` and contain the numerical value of the argument.

When called, `filter_function` must return a tuple `(operation, return_value)`. `operation` can either be a string description of the operation that was prevented by the filter, to be printed after the process terminates, or `None`, in which case nothing will be printed. `return_value` can either be a numerical value, in which case the syscall invocation will be prevented and the return value received by the caller will be set to that value, or `None`, in which case the invocation will be allowed to proceed as normal.

### Example

Here, `maybenot`'s plugin API is used to implement an exotic type of access control: Restricting read access based on the *content* of the file in question. If a file being opened for reading contains the word **SECRET**, the plugin blocks the `open`/`openat` syscall and returns an error.

#### `read_secret_file.py`

```python
from os import O_WRONLY
from os.path import isfile
from maybenot import T, register_filter

def filter_open(path, flags):
    if path.startswith("/home/") and isfile(path) and not (flags & O_WRONLY):
        with open(path, "r") as f:
            if "SECRET" in f.read():
                return "%s %s" % (T.red("read secret file"), T.underline(path)), -1
            else:
                return None, None
    else:
        return None, None

register_filter("open", lambda process, args:
                filter_open(process.full_path(args[0]), args[1]))
register_filter("openat", lambda process, args:
                filter_open(process.full_path(args[1], args[0]), args[2]))
```

Indeed, the plugin works as expected:

```
[user@localhost]$ maybenot --plugin read_secret_file.py --deny read_secret_file -- bash
$ echo "This is a normal file." > file_1
$ echo "This is a SECRET file." > file_2
$ cat file_1
This is a normal file.
$ cat file_2
cat: file_2: Operation not permitted
```


## License

Copyright &copy; 2016-2017 Philipp Emanuel Weidmann (<pew@worldwidemann.com>)
Copyright &copy; 2023 Rinat Sabitov (<rinat.sabitov@gmail.com>)

Released under the terms of the [GNU General Public License, version 3](https://gnu.org/licenses/gpl.html)

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/histrio/maybenot",
    "name": "maybenot",
    "maintainer": null,
    "docs_url": null,
    "requires_python": "<4.0,>=3.8",
    "maintainer_email": null,
    "keywords": "command preview, system calls interception, file system operations simulation, dry run utility",
    "author": "Philipp Emanuel Weidmann",
    "author_email": "pew@worldwidemann.com",
    "download_url": "https://files.pythonhosted.org/packages/84/00/e07c67052a29a45d5f1ffde6ba6ffb41f1f7e73dc1ba849b43fd08dca399/maybenot-0.5.0rc1.tar.gz",
    "platform": null,
    "description": "[![Release](https://img.shields.io/github/v/release/histrio/maybenot)](https://img.shields.io/github/v/release/histrio/maybenot)\n[![Tests](https://github.com/histrio/maybenot/actions/workflows/main.yml/badge.svg?branch=master)](https://github.com/histrio/maybenot/actions/workflows/main.yml)\n[![codecov](https://codecov.io/gh/histrio/maybenot/graph/badge.svg?token=X6ZMMZNN5T)](https://codecov.io/gh/histrio/maybenot)\n[![Commit activity](https://img.shields.io/github/commit-activity/m/histrio/maybenot)](https://img.shields.io/github/commit-activity/m/histrio/maybenot)\n[![License](https://img.shields.io/github/license/histrio/maybenot)](https://img.shields.io/github/license/histrio/maybenot)\n\n\n---\n\n\n```\nrm -rf pic*\n```\n\nAre you sure? Are you *one hundred percent* sure?\n\n\n# `maybenot`...\n\n... allows you to run a command and see what it does to your files *without actually doing it!* After reviewing the operations listed, you can then decide whether you really want these things to happen or not.\n\n![Screenshot](screenshot.png)\n\n\n## What is this sorcery?!?\n\n`maybenot` runs processes under the control of [ptrace](https://en.wikipedia.org/wiki/Ptrace) (with the help of the excellent [python-ptrace](https://github.com/haypo/python-ptrace) library). When it intercepts a system call that is about to make changes to the file system, it logs that call, and then modifies CPU registers to both redirect the call to an invalid syscall ID (effectively turning it into a no-op) and set the return value of that no-op call to one indicating success of the original call.\n\nAs a result, the process believes that everything it is trying to do is actually happening, when in reality nothing is.\n\nThat being said, `maybenot` **should :warning: NEVER :warning: be used to run untrusted code** on a system you care about! A process running under `maybenot` can still do serious damage to your system because only a handful of syscalls are blocked. It can also check whether an operation such as deleting a file succeeded using read-only syscalls, and *alter its behavior accordingly.* Therefore, a rerun without restrictions is not guaranteed to always produce the displayed operations.\n\nCurrently, `maybenot` is best thought of as an (alpha-quality) \"what exactly will this command I typed myself do?\" tool.\n\n\n## Installation\n\n`maybenot` runs on Linux :penguin: and requires [Python 3.8+](https://www.python.org/) :snake:. If you have the [pip](https://pip.pypa.io) package manager, all you need to do is run\n\n```\npip install maybe\n```\n\neither as a superuser or from a [virtualenv](https://virtualenv.pypa.io) environment. To develop `maybenot`, clone the repository and run\n\n```\npip install -e .\n```\n\nin its main directory to install the package in editable mode.\n\n\n## Usage\n\n```\nmaybenot [options] command [argument ...]\n```\n\n### Positional arguments\n\n| Argument | Description |\n| --- | --- |\n| `command` | the command to run under `maybenot`'s control |\n| `argument ...` | argument(s) to pass to `command` |\n\n### Optional arguments\n\n| Argument | Description |\n| --- | --- |\n| `-a OPERATION ...`,<br>`--allow OPERATION ...`<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; | allow the command to perform the specified operation(s). all other operations will be denied. possible values for `OPERATION` are: `change_owner`, `change_permissions`, `create_directory`, `create_link`, `create_write_file`, `delete`, `move`; as well as any filter scopes defined by loaded plugins |\n| `-d OPERATION ...`,<br>`--deny OPERATION ...` | deny the command the specified operation(s). all other operations will be allowed. see `--allow` for a list of possible values for `OPERATION`. `--allow` and `--deny` cannot be combined |\n| `-p FILE ...`,<br>`--plugin FILE ...` | load the specified [plugin](#plugin-api) script(s) |\n| `-l`, `--list-only` | list operations without header, indentation and rerun prompt |\n| `--style-output {yes,no,auto}` | colorize output using ANSI escape sequences (`yes`/`no`) or automatically decide based on whether stdout is a terminal (`auto`, default) |\n| `-v`, `--verbose` | if specified once, print every filtered syscall. if specified twice, print every syscall, highlighting filtered syscalls |\n| `--version` | show program's version number and exit |\n| `-h`, `--help` | show a help message and exit |\n\n\n## Plugin API\n\nBy default, `maybenot` intercepts and blocks all syscalls that can make permanent modifications to the system. For more specialized syscall filtering needs, `maybenot` provides a simple yet powerful plugin API. Filter plugins are written in pure Python and use the same interfaces as [`maybenot`'s built-in filters](maybenot/filters).\n\nThe public API is composed of the following two members:\n\n#### *maybenot.*`T`\n\nA [Blessings](https://github.com/erikrose/blessings) `Terminal` object that can be used to format console output (such as `operation` as documented below). Output formatted with this object automatically complies with the `--style-output` command line argument.\n\n#### *maybenot.*`register_filter(syscall, filter_function, filter_scope=None)`\n\nAdd the filter `filter_function` to the filter registry. If the filter is enabled (which is the default, but can be altered with the `--allow` and `--deny` command line arguments), it intercepts all calls to `syscall` made by the controlled process. `filter_scope` determines the key to be used in conjunction with `--allow` and `--deny` to enable/disable the filter (multiple filters can share the same key). If `filter_scope` is omitted or `None`, the last part of the plugin's module name is used.\n\n`filter_function` itself must conform to the signature `filter_function(process, args)`. `process` is a [`Process`](maybenot/process.py) control object that can be used to inspect and manipulate the process, while `args` is the list of arguments passed to the syscall in the order in which they appear in the syscall's signature. If an argument represents a (pointer to a) filename, the argument will be of type `str` and contain the filename, otherwise it will be of type `int` and contain the numerical value of the argument.\n\nWhen called, `filter_function` must return a tuple `(operation, return_value)`. `operation` can either be a string description of the operation that was prevented by the filter, to be printed after the process terminates, or `None`, in which case nothing will be printed. `return_value` can either be a numerical value, in which case the syscall invocation will be prevented and the return value received by the caller will be set to that value, or `None`, in which case the invocation will be allowed to proceed as normal.\n\n### Example\n\nHere, `maybenot`'s plugin API is used to implement an exotic type of access control: Restricting read access based on the *content* of the file in question. If a file being opened for reading contains the word **SECRET**, the plugin blocks the `open`/`openat` syscall and returns an error.\n\n#### `read_secret_file.py`\n\n```python\nfrom os import O_WRONLY\nfrom os.path import isfile\nfrom maybenot import T, register_filter\n\ndef filter_open(path, flags):\n    if path.startswith(\"/home/\") and isfile(path) and not (flags & O_WRONLY):\n        with open(path, \"r\") as f:\n            if \"SECRET\" in f.read():\n                return \"%s %s\" % (T.red(\"read secret file\"), T.underline(path)), -1\n            else:\n                return None, None\n    else:\n        return None, None\n\nregister_filter(\"open\", lambda process, args:\n                filter_open(process.full_path(args[0]), args[1]))\nregister_filter(\"openat\", lambda process, args:\n                filter_open(process.full_path(args[1], args[0]), args[2]))\n```\n\nIndeed, the plugin works as expected:\n\n```\n[user@localhost]$ maybenot --plugin read_secret_file.py --deny read_secret_file -- bash\n$ echo \"This is a normal file.\" > file_1\n$ echo \"This is a SECRET file.\" > file_2\n$ cat file_1\nThis is a normal file.\n$ cat file_2\ncat: file_2: Operation not permitted\n```\n\n\n## License\n\nCopyright &copy; 2016-2017 Philipp Emanuel Weidmann (<pew@worldwidemann.com>)\nCopyright &copy; 2023 Rinat Sabitov (<rinat.sabitov@gmail.com>)\n\nReleased under the terms of the [GNU General Public License, version 3](https://gnu.org/licenses/gpl.html)\n",
    "bugtrack_url": null,
    "license": "GPLv3",
    "summary": "See what a program does before deciding whether you really want it to happen.",
    "version": "0.5.0rc1",
    "project_urls": {
        "Documentation": "https://histrio.github.io/maybenot/",
        "Homepage": "https://github.com/histrio/maybenot",
        "Repository": "https://github.com/histrio/maybenot"
    },
    "split_keywords": [
        "command preview",
        " system calls interception",
        " file system operations simulation",
        " dry run utility"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "fa51af569cdeaa9b5c8b801dcfd9b14caf27cbeaf17bdf6def7c0d474e8708f7",
                "md5": "c50e37dbbcc03d81c1d50286fb6dee82",
                "sha256": "d7a12a41dffe43024550b692d7c640a9a3bfda4b97593d9fa366fda346882c47"
            },
            "downloads": -1,
            "filename": "maybenot-0.5.0rc1-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "c50e37dbbcc03d81c1d50286fb6dee82",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": "<4.0,>=3.8",
            "size": 28241,
            "upload_time": "2024-04-10T08:23:35",
            "upload_time_iso_8601": "2024-04-10T08:23:35.865733Z",
            "url": "https://files.pythonhosted.org/packages/fa/51/af569cdeaa9b5c8b801dcfd9b14caf27cbeaf17bdf6def7c0d474e8708f7/maybenot-0.5.0rc1-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8400e07c67052a29a45d5f1ffde6ba6ffb41f1f7e73dc1ba849b43fd08dca399",
                "md5": "2e18b0444884571a2f0c91a8237fddd4",
                "sha256": "17faad4dd96d57ae49cf7a84a72d322818babe3405b90d38709912c77cca5f89"
            },
            "downloads": -1,
            "filename": "maybenot-0.5.0rc1.tar.gz",
            "has_sig": false,
            "md5_digest": "2e18b0444884571a2f0c91a8237fddd4",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": "<4.0,>=3.8",
            "size": 25818,
            "upload_time": "2024-04-10T08:23:38",
            "upload_time_iso_8601": "2024-04-10T08:23:38.048085Z",
            "url": "https://files.pythonhosted.org/packages/84/00/e07c67052a29a45d5f1ffde6ba6ffb41f1f7e73dc1ba849b43fd08dca399/maybenot-0.5.0rc1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-04-10 08:23:38",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "histrio",
    "github_project": "maybenot",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "tox": true,
    "lcname": "maybenot"
}
        
Elapsed time: 4.10129s