named_semaphores


Namenamed_semaphores JSON
Version 1.0.3 PyPI version JSON
download
home_pageNone
SummaryHigh-level POSIX IPC named semaphores for Python
upload_time2024-11-25 08:42:38
maintainerNone
docs_urlNone
authorNone
requires_python>=3.8
licenseNone
keywords concurrency inter-process communication ipc named-semaphore named-semaphores posix posix-ipc semaphore semaphores synchronization threading
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # Pythonic API for POSIX IPC Named Semaphores

![License](https://img.shields.io/github/license/johacks/named-semaphores)
![PyPI Version](https://img.shields.io/pypi/v/named-semaphores)
![Dependencies](https://img.shields.io/librariesio/release/pypi/named-semaphores)
[![Coverage Status](https://codecov.io/gh/johacks/named-semaphores/branch/main/graph/badge.svg)](https://codecov.io/gh/johacks/named-semaphores)
[![Codacy Badge](https://app.codacy.com/project/badge/Grade/0c27d0f8d27a468598813292178b4881)](https://app.codacy.com/gh/johacks/named-semaphores/dashboard?utm_source=gh&utm_medium=referral&utm_content=&utm_campaign=Badge_grade)

## Description

Use POSIX IPC named semaphores with a high-level API similar to Python's `threading.Semaphore`. This repository wraps around existing bindings from the [posix_ipc](https://pypi.org/project/posix-ipc/) package to provide a more Pythonic interface.

## Features

- **Named Semaphores**: Handle POSIX IPC named semaphores with ease.
- **Cross-Platform**: Works on Linux, macOS, and Windows (via Cygwin).
- **Thread-Safe**: Built on POSIX IPC, ensuring robust multi-process handling.
- **Pythonic API**: Similar to Python's built-in `threading.Semaphore` for familiarity.
- **Flexible Creation**: Choose how to handle existing semaphores (`RAISE_IF_EXISTS`, `LINK_OR_CREATE`, etc.).
- **Timeouts**: Optionally specify timeouts for acquiring semaphores (platform-dependent).
- **Automatic Cleanup**: Semaphore can be automatically unlinked when the object is deleted.

## Installation

You can install the package from PyPI:

```bash
pip install named-semaphores
```

## Usage

The `NamedSemaphore` class provides a high-level API for working with POSIX IPC named semaphores. Below are various usage examples demonstrating its flexibility and functionality.

### 1. **Basic Semaphore Creation**

Create a semaphore with a default initial value of 1.

```python
from named_semaphores import NamedSemaphore

sem = NamedSemaphore("example_semaphore")
print(f"Semaphore '{sem.name}' created.")
```

You can also specify a custom initial value for the semaphore.

```python
sem = NamedSemaphore("example_semaphore", initial_value=3)
print(f"Semaphore '{sem.name}' created with initial value of 3.")
```

---

### 2. **Acquire, Release, and Timeouts**

You can acquire and release the semaphore, specify a custom initial value, or use a timeout for acquiring (if supported on your platform).

#### Acquire and Release

```python
sem = NamedSemaphore("example_semaphore")

# Acquire the semaphore
sem.acquire()
try:
    print("Critical section protected by semaphore.")
finally:
    # Release the semaphore
    sem.release()
    print("Semaphore released.")
```

#### Acquire with Timeout

```python
# Acquire the semaphore with a 5-second timeout
acquired = sem.acquire(timeout=5)
if acquired:
    print("Semaphore acquired within timeout.")
    sem.release()
else:
    print("Failed to acquire semaphore within timeout.")
```

---

### 3. **Using as a Context Manager**
The semaphore can be used in a with statement to automatically acquire and release it.

```python
with NamedSemaphore("example_semaphore") as sem:
    print(f"Semaphore '{sem.name}' acquired.")
    # Critical section here
print(f"Semaphore '{sem.name}' released.")
```

---

### 4. **Unlinking a Semaphore**

Unlinking a semaphore removes it from the system after all processes using it have closed it. Note some comments made by the `posix_ipc` author:

```markdown
“If any processes have the semaphore open when unlink is called, the call to unlink returns immediately but destruction of the semaphore is postponed until all processes have closed the semaphore.

Note, however, that once a semaphore has been unlinked, calls to open() with the same name should refer to a new semaphore. Sound confusing? It is, and you'd probably be wise structure your code so as to avoid this situation.”
```
By default, semaphores created by this class are automatically unlinked when the object is deleted. You can override this behavior with the `unlink_on_delete` parameter.

- Automatic unlinking example:

```python
# Create a semaphore
handle_create = NamedSemaphore(
    "example_semaphore",
    handle_existence=NamedSemaphore.Flags.RAISE_IF_EXISTS
)
# Link to the semaphore
handle_link = NamedSemaphore(
    "example_semaphore",
    handle_existence=NamedSemaphore.Flags.RAISE_IF_NOT_EXISTS
)
# Trigger garbage collection
del handle_link  # Only closes the handle, does not unlink the semaphore
del handle_create  # Unlinks the semaphore, because this handle created it
```
- Manual unlinking example:

```python
# Set `unlink_on_delete` to False to prevent automatic unlinking on object deletion
sem = NamedSemaphore("example_semaphore", unlink_on_delete=False)
del sem  # Semaphore is not unlinked
sem = NamedSemaphore("example_semaphore", unlink_on_delete=False)
# Unlink the semaphore manually, handle will be closed in the destructor
sem.unlink()
```

---

### 5. **Handling of Existing Semaphores**

The `NamedSemaphore` class provides flags to handle existing semaphores in different ways. The safest ways are the `RAISE_IF_EXISTS` and `RAISE_IF_NOT_EXISTS` flags, which raise an error if the semaphore already exists or does not exist, respectively.

Some other flags are `LINK_OR_CREATE` and `UNLINK_AND_CREATE`, which link to an existing semaphore or unlink it if it already exists. They may be useful in certain scenarios, but should be used with caution, as they may silently hide issues in the code (e.g., not properly cleaning up semaphores).

#### Raise an error if the semaphore already exists

Use the `RAISE_IF_EXISTS` flag to ensure a new semaphore is created and raise an error if one already exists.

```python
try:
    sem = NamedSemaphore(
        "example_semaphore",
        handle_existence=NamedSemaphore.Flags.RAISE_IF_EXISTS
    )
    print(f"Semaphore '{sem.name}' created without existing conflict.")
except FileExistsError:
    print("Handling existing semaphore error...")
```

#### Link to a semaphore or create it if it doesn't exist

Use the `LINK_OR_CREATE` flag to link to an existing semaphore or create a new one if it doesn't exist.

```python
sem = NamedSemaphore(
    "example_semaphore",
    handle_existence=NamedSemaphore.Flags.LINK_OR_CREATE
)
print(f"Semaphore '{sem.name}' linked or created.")
```

#### Link to a semaphore or raise an error if it doesn't exist

Use the `RAISE_IF_NOT_EXISTS` flag to link to an existing semaphore, raising an error if it does not exist.

```python
try:
    sem = NamedSemaphore(
        "example_semaphore",
        handle_existence=NamedSemaphore.Flags.RAISE_IF_NOT_EXISTS
    )
    print(f"Semaphore '{sem.name}' linked.")
except FileNotFoundError:
    print("Handling non-existing semaphore error...")
```

#### Create a Semaphore, unlinking it if it already exists

Use the `UNLINK_AND_CREATE` flag to unlink semaphore if it already exists and create a new one.

```python
sem = NamedSemaphore(
    "example_semaphore",
    handle_existence=NamedSemaphore.Flags.UNLINK_AND_CREATE
)
print(f"Semaphore '{sem.name}' deleted and created.")
```

**Note**: although convenient, rather than using a forced deletion and create, it is better practice to handle the existence of semaphores in a more controlled manner. An already existing semaphore usually indicates that it is being used by another process, or that it was not properly cleaned up by a previous process.

---

These examples cover the most common scenarios for using NamedSemaphore. For advanced use cases or troubleshooting, refer to the module's documentation or docstrings.

## Contributing

Contributions are welcome! Please follow these steps:

1. Fork the repository.
2. Create a new branch (`git checkout -b feature/your-feature`).
3. Commit your changes (`git commit -m "Add your feature"`).
4. Push to the branch (`git push origin feature/your-feature`).
5. Open a Pull Request.

## License

This project is licensed under the Apache License 2.0. See the [LICENSE](LICENSE) file for more details.

Please note that this project depends on the [posix_ipc](https://pypi.org/project/posix-ipc/) package, which is licensed under a BSD-style license. For details about the `posix_ipc` license, refer to its [license file](https://raw.githubusercontent.com/osvenskan/posix_ipc/4db678001be2f16175c70cb88d4fb9f9126333f5/LICENSE).

            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "named_semaphores",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.8",
    "maintainer_email": "johacks <joaquinjimenezlc@gmail.com>",
    "keywords": "concurrency, inter-process communication, ipc, named-semaphore, named-semaphores, posix, posix-ipc, semaphore, semaphores, synchronization, threading",
    "author": null,
    "author_email": "johacks <joaquinjimenezlc@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/17/9a/2c0756ad9f0c12574175433ff2b291ff4c778633acafde171e9b7851dd6f/named_semaphores-1.0.3.tar.gz",
    "platform": null,
    "description": "# Pythonic API for POSIX IPC Named Semaphores\n\n![License](https://img.shields.io/github/license/johacks/named-semaphores)\n![PyPI Version](https://img.shields.io/pypi/v/named-semaphores)\n![Dependencies](https://img.shields.io/librariesio/release/pypi/named-semaphores)\n[![Coverage Status](https://codecov.io/gh/johacks/named-semaphores/branch/main/graph/badge.svg)](https://codecov.io/gh/johacks/named-semaphores)\n[![Codacy Badge](https://app.codacy.com/project/badge/Grade/0c27d0f8d27a468598813292178b4881)](https://app.codacy.com/gh/johacks/named-semaphores/dashboard?utm_source=gh&utm_medium=referral&utm_content=&utm_campaign=Badge_grade)\n\n## Description\n\nUse POSIX IPC named semaphores with a high-level API similar to Python's `threading.Semaphore`. This repository wraps around existing bindings from the [posix_ipc](https://pypi.org/project/posix-ipc/) package to provide a more Pythonic interface.\n\n## Features\n\n- **Named Semaphores**: Handle POSIX IPC named semaphores with ease.\n- **Cross-Platform**: Works on Linux, macOS, and Windows (via Cygwin).\n- **Thread-Safe**: Built on POSIX IPC, ensuring robust multi-process handling.\n- **Pythonic API**: Similar to Python's built-in `threading.Semaphore` for familiarity.\n- **Flexible Creation**: Choose how to handle existing semaphores (`RAISE_IF_EXISTS`, `LINK_OR_CREATE`, etc.).\n- **Timeouts**: Optionally specify timeouts for acquiring semaphores (platform-dependent).\n- **Automatic Cleanup**: Semaphore can be automatically unlinked when the object is deleted.\n\n## Installation\n\nYou can install the package from PyPI:\n\n```bash\npip install named-semaphores\n```\n\n## Usage\n\nThe `NamedSemaphore` class provides a high-level API for working with POSIX IPC named semaphores. Below are various usage examples demonstrating its flexibility and functionality.\n\n### 1. **Basic Semaphore Creation**\n\nCreate a semaphore with a default initial value of 1.\n\n```python\nfrom named_semaphores import NamedSemaphore\n\nsem = NamedSemaphore(\"example_semaphore\")\nprint(f\"Semaphore '{sem.name}' created.\")\n```\n\nYou can also specify a custom initial value for the semaphore.\n\n```python\nsem = NamedSemaphore(\"example_semaphore\", initial_value=3)\nprint(f\"Semaphore '{sem.name}' created with initial value of 3.\")\n```\n\n---\n\n### 2. **Acquire, Release, and Timeouts**\n\nYou can acquire and release the semaphore, specify a custom initial value, or use a timeout for acquiring (if supported on your platform).\n\n#### Acquire and Release\n\n```python\nsem = NamedSemaphore(\"example_semaphore\")\n\n# Acquire the semaphore\nsem.acquire()\ntry:\n    print(\"Critical section protected by semaphore.\")\nfinally:\n    # Release the semaphore\n    sem.release()\n    print(\"Semaphore released.\")\n```\n\n#### Acquire with Timeout\n\n```python\n# Acquire the semaphore with a 5-second timeout\nacquired = sem.acquire(timeout=5)\nif acquired:\n    print(\"Semaphore acquired within timeout.\")\n    sem.release()\nelse:\n    print(\"Failed to acquire semaphore within timeout.\")\n```\n\n---\n\n### 3. **Using as a Context Manager**\nThe semaphore can be used in a with statement to automatically acquire and release it.\n\n```python\nwith NamedSemaphore(\"example_semaphore\") as sem:\n    print(f\"Semaphore '{sem.name}' acquired.\")\n    # Critical section here\nprint(f\"Semaphore '{sem.name}' released.\")\n```\n\n---\n\n### 4. **Unlinking a Semaphore**\n\nUnlinking a semaphore removes it from the system after all processes using it have closed it. Note some comments made by the `posix_ipc` author:\n\n```markdown\n\u201cIf any processes have the semaphore open when unlink is called, the call to unlink returns immediately but destruction of the semaphore is postponed until all processes have closed the semaphore.\n\nNote, however, that once a semaphore has been unlinked, calls to open() with the same name should refer to a new semaphore. Sound confusing? It is, and you'd probably be wise structure your code so as to avoid this situation.\u201d\n```\nBy default, semaphores created by this class are automatically unlinked when the object is deleted. You can override this behavior with the `unlink_on_delete` parameter.\n\n- Automatic unlinking example:\n\n```python\n# Create a semaphore\nhandle_create = NamedSemaphore(\n    \"example_semaphore\",\n    handle_existence=NamedSemaphore.Flags.RAISE_IF_EXISTS\n)\n# Link to the semaphore\nhandle_link = NamedSemaphore(\n    \"example_semaphore\",\n    handle_existence=NamedSemaphore.Flags.RAISE_IF_NOT_EXISTS\n)\n# Trigger garbage collection\ndel handle_link  # Only closes the handle, does not unlink the semaphore\ndel handle_create  # Unlinks the semaphore, because this handle created it\n```\n- Manual unlinking example:\n\n```python\n# Set `unlink_on_delete` to False to prevent automatic unlinking on object deletion\nsem = NamedSemaphore(\"example_semaphore\", unlink_on_delete=False)\ndel sem  # Semaphore is not unlinked\nsem = NamedSemaphore(\"example_semaphore\", unlink_on_delete=False)\n# Unlink the semaphore manually, handle will be closed in the destructor\nsem.unlink()\n```\n\n---\n\n### 5. **Handling of Existing Semaphores**\n\nThe `NamedSemaphore` class provides flags to handle existing semaphores in different ways. The safest ways are the `RAISE_IF_EXISTS` and `RAISE_IF_NOT_EXISTS` flags, which raise an error if the semaphore already exists or does not exist, respectively.\n\nSome other flags are `LINK_OR_CREATE` and `UNLINK_AND_CREATE`, which link to an existing semaphore or unlink it if it already exists. They may be useful in certain scenarios, but should be used with caution, as they may silently hide issues in the code (e.g., not properly cleaning up semaphores).\n\n#### Raise an error if the semaphore already exists\n\nUse the `RAISE_IF_EXISTS` flag to ensure a new semaphore is created and raise an error if one already exists.\n\n```python\ntry:\n    sem = NamedSemaphore(\n        \"example_semaphore\",\n        handle_existence=NamedSemaphore.Flags.RAISE_IF_EXISTS\n    )\n    print(f\"Semaphore '{sem.name}' created without existing conflict.\")\nexcept FileExistsError:\n    print(\"Handling existing semaphore error...\")\n```\n\n#### Link to a semaphore or create it if it doesn't exist\n\nUse the `LINK_OR_CREATE` flag to link to an existing semaphore or create a new one if it doesn't exist.\n\n```python\nsem = NamedSemaphore(\n    \"example_semaphore\",\n    handle_existence=NamedSemaphore.Flags.LINK_OR_CREATE\n)\nprint(f\"Semaphore '{sem.name}' linked or created.\")\n```\n\n#### Link to a semaphore or raise an error if it doesn't exist\n\nUse the `RAISE_IF_NOT_EXISTS` flag to link to an existing semaphore, raising an error if it does not exist.\n\n```python\ntry:\n    sem = NamedSemaphore(\n        \"example_semaphore\",\n        handle_existence=NamedSemaphore.Flags.RAISE_IF_NOT_EXISTS\n    )\n    print(f\"Semaphore '{sem.name}' linked.\")\nexcept FileNotFoundError:\n    print(\"Handling non-existing semaphore error...\")\n```\n\n#### Create a Semaphore, unlinking it if it already exists\n\nUse the `UNLINK_AND_CREATE` flag to unlink semaphore if it already exists and create a new one.\n\n```python\nsem = NamedSemaphore(\n    \"example_semaphore\",\n    handle_existence=NamedSemaphore.Flags.UNLINK_AND_CREATE\n)\nprint(f\"Semaphore '{sem.name}' deleted and created.\")\n```\n\n**Note**: although convenient, rather than using a forced deletion and create, it is better practice to handle the existence of semaphores in a more controlled manner. An already existing semaphore usually indicates that it is being used by another process, or that it was not properly cleaned up by a previous process.\n\n---\n\nThese examples cover the most common scenarios for using NamedSemaphore. For advanced use cases or troubleshooting, refer to the module's documentation or docstrings.\n\n## Contributing\n\nContributions are welcome! Please follow these steps:\n\n1. Fork the repository.\n2. Create a new branch (`git checkout -b feature/your-feature`).\n3. Commit your changes (`git commit -m \"Add your feature\"`).\n4. Push to the branch (`git push origin feature/your-feature`).\n5. Open a Pull Request.\n\n## License\n\nThis project is licensed under the Apache License 2.0. See the [LICENSE](LICENSE) file for more details.\n\nPlease note that this project depends on the [posix_ipc](https://pypi.org/project/posix-ipc/) package, which is licensed under a BSD-style license. For details about the `posix_ipc` license, refer to its [license file](https://raw.githubusercontent.com/osvenskan/posix_ipc/4db678001be2f16175c70cb88d4fb9f9126333f5/LICENSE).\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "High-level POSIX IPC named semaphores for Python",
    "version": "1.0.3",
    "project_urls": {
        "Issues": "https://github.com/johacks/named-semaphores/issues",
        "Repository": "https://github.com/johacks/named-semaphores"
    },
    "split_keywords": [
        "concurrency",
        " inter-process communication",
        " ipc",
        " named-semaphore",
        " named-semaphores",
        " posix",
        " posix-ipc",
        " semaphore",
        " semaphores",
        " synchronization",
        " threading"
    ],
    "urls": [
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "b7abaed09096024ee8d35b376919a4fdd734bb0ae4b5c1416221218f4e21a9e5",
                "md5": "f6edbdef947fc61865af0bce9403e18d",
                "sha256": "d003336ab4003585cd800b07701d0fd947ec15dca34454dbab69e59bd780f0f3"
            },
            "downloads": -1,
            "filename": "named_semaphores-1.0.3-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "f6edbdef947fc61865af0bce9403e18d",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.8",
            "size": 12997,
            "upload_time": "2024-11-25T08:42:37",
            "upload_time_iso_8601": "2024-11-25T08:42:37.261358Z",
            "url": "https://files.pythonhosted.org/packages/b7/ab/aed09096024ee8d35b376919a4fdd734bb0ae4b5c1416221218f4e21a9e5/named_semaphores-1.0.3-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "179a2c0756ad9f0c12574175433ff2b291ff4c778633acafde171e9b7851dd6f",
                "md5": "6c5c9498a307f57611f0e70bb7fd0a12",
                "sha256": "e3022696d73e70ca1715dcfcde9f72cf7c75d65c664330c16c8f4afb33390996"
            },
            "downloads": -1,
            "filename": "named_semaphores-1.0.3.tar.gz",
            "has_sig": false,
            "md5_digest": "6c5c9498a307f57611f0e70bb7fd0a12",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.8",
            "size": 25860,
            "upload_time": "2024-11-25T08:42:38",
            "upload_time_iso_8601": "2024-11-25T08:42:38.409976Z",
            "url": "https://files.pythonhosted.org/packages/17/9a/2c0756ad9f0c12574175433ff2b291ff4c778633acafde171e9b7851dd6f/named_semaphores-1.0.3.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-11-25 08:42:38",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "johacks",
    "github_project": "named-semaphores",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "named_semaphores"
}
        
Elapsed time: 0.36917s