Metadata-Version: 2.4
Name: plankapy
Version: 2.3.2
Summary: A Python interface for Plankanban
Author-email: Hayden Welch <hwelch@finelines-engineering.com>
Project-URL: Homepage, https://github.com/hwelch-fle/plankapy
Project-URL: Documentation, https://hwelch-fle.github.io/plankapy/plankapy.html
Project-URL: Issues, https://github.com/hwelch-fle/plankapy/issues
Project-URL: Discussions, https://github.com/hwelch-fle/plankapy/discussions
Classifier: Programming Language :: Python :: 3
Classifier: License :: OSI Approved :: GNU Affero General Public License v3 or later (AGPLv3+)
Classifier: Operating System :: OS Independent
Requires-Python: >=3.13
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: httpx>=0.28.1
Dynamic: license-file

# plankapy
A Python API for [Planka](https://github.com/plankanban/planka)

![PyPI - License](https://img.shields.io/pypi/l/plankapy)

![PyPI - Version](https://img.shields.io/pypi/v/plankapy) ![PyPI - Downloads](https://img.shields.io/pypi/dw/plankapy) ![GitHub last commit](https://img.shields.io/github/last-commit/hwelch-fle/plankapy)

[![Release](https://github.com/hwelch-fle/plankapy/actions/workflows/python-publish.yml/badge.svg)](https://github.com/hwelch-fle/plankapy/actions/workflows/python-publish.yml) [![Docs](https://github.com/hwelch-fle/plankapy/actions/workflows/docs.yml/badge.svg)](https://github.com/hwelch-fle/plankapy/actions/workflows/docs.yml) 


## Installation
```bash
pip install plankapy
```

## Documentation
The full documentation can be found [here](https://hwelch-fle.github.io/plankapy/).

## Getting Started
After pip installing, you can connect to a Planka instance using a Planka interface object

```python
from plankapy.v2 import Planka

planka = Planka('https://planka.mydomain.com')
```
After creating the Planka instance, you can authenticate using the `login` method:
```python
planka.login(username='username', password='password')
-- OR --
planka.login(api_key='MY_API_KEY')
```

Logging in for the first time required accepting the instance ToS. Consult with your instance admin on avalable ToS localizations and required ToS types for your user.
```python
planka.login(username=..., password=..., accept_terms=True)

-- With Language --
planka.login(username=..., password=..., accept_terms=True, lang='ja-JP')

-- Specific ToS --
planka.login(username=..., password=..., accept_terms=True, terms_type='extended')
```
Now that you've authenticated with the server, you have access to everything that you'd be able to access through the web-ui

```python
>>> planka.me
User({...})
>>> planka.projects
[Project({...}), Project({...}), ...]
```

## Features

### Fully Typed
Interfaces for all planka objects are fully documented and typed in the source, so if you are using an IDE that supports a python language server you will get full autocompletion.
![autocomplete](/res/readme-autocomplete.png)

All methods are fully documented using doctrings and contain all info needed to properly form a function call:
![inline_doc](/res/readme-inline_doc.png)

Since all objects are initialized as a series of endpoints with a local `schema` cache, it allows you to access any object from any other object in the tree without ever losing track of what you have:
![circular](/res/readme-circular.png)

All endpoints and schemas are documeted using `TypedDict` syntax, so you can safely create raw API calls using the `PlankaEndpoints` object:
![typed-endpoints](/res/readme-PlankaEndpoints.png)

All responses are fully typed as well:
!['typed-response-1](/res/readme-typed_response_1.png)
!['typed-response-2](/res/readme-typed_response_2.png)
!['typed-response-3](/res/readme-typed_response_3.png)


### NOTE:
When using plankapy it is reccommended to turn your type checker settings to the strictest mode possible. Since most interfaces forego runtime checks for extensive type hints. 

### Model Lists
When reading the published docs, all property attributes that are typed as `list[<PlankaModel[T]>]` will be converted to `ModelList` at runtime. This list subclass can be used like a regulat list, bu allows indexing and filtering using `__getitem__` syntax

#### `int/slice`
Since `ModelList` subclasses list, all regular list interfaces still work:
```python
>>> board.cards[-1]
Card({...})
>>> board.cards[:2]
[Card({...}), Card({})]
```
#### `str/Id`
If a string is used to index a model list, the last model with the matching id is popped. If the id does not exist, the exception raised will be the same as popping from an empty list
```python
>>> my_card_id = '...'
>>> board.cards[my_card_id]
Card({'id': '...'})
```
#### `SchemaFilter/dict`
If you want to define a filter on the model schema, you can pass a dictionary of schema values or expressions to the index:
```python
>>> board.cards[{'name': 'My Card'}]
[Card({'name': 'My Card'})]

>>> board.cards[{'name': lambda name: 'My' in name}]
[Card({'name': 'My Card'}), Card({'name': 'My Other Card'})]

```
#### `ModelFilter`
You can also specify a raw expression to filter the list:
```python
>>> board.cards[lambda card: planka.me in card.members]
``` 
#### `Model`
You can also direcly pass a Model to the index. The returned result is another ModelList that can be `dpop`'ed to check for membership
```python
>>> board.users[planka.me].dpop()
None # Not in board
```

#### `extract`
Model lists also allow schema extraction using the `extract` method. If a single key is passed, a list of values is returned. If multiple keys are passed, a list of tuples of values is returned
```python
>>> board.cards.extract('name')
['My Card', 'My Other Card']
>>> board.card.extract('name', 'dueDate')
[('My Card', None), ('My Other Card', '2026...')]
```

#### `dpop`
A way to safely pop from the model list. Has the same interface as `list.pop` but accepts a `default` keyword argument which is det to `None` when not set by the user:
```python
>>> board.lists[0].pop(0)
IndexError...
>>> board.lists[0].dpop(0, 'Empty')
'Empty'
``` 

### Live Updates
Every model contains a `schema` cache attribute that is used for direct attribute access. This means that checking the value of `name` for a card will not change until `sync` is called
```python
>>> card = board.cards[{'name': 'My Card'}].dpop()
>>> card.name
'My Card'

# Change Name on server to 'Not My Card'
>>> card.name
'My Card'
>>> card.sync()
>>> card.name
'Not My Card'
```
If you want your functions to maintain sync with the server, you access the card via other endpoints. All properties that return item lists will re-construct themselves from the response when accessed

```python
>>> card = lambda: board.cards[0]
>>> card().name
'My Card'

# Change Name on server to 'Not My Card'
>>> card().name
'Not My Card'
```
This example aliases a request chain to a callable lambda function. You can also just use the full desired request path too
```python
>>> board.cards[0].name
'My Card'

# Change Name on server to 'Not My Card'
>>> board.cards[0].name
'Not My Card'
```
NOTE: This specific structure is written to always get the *first* card in the board. This could be useful, but you will usually want to apply a specific filter. 

### Attribute access in Loops
Since extended calls can fire off multiple requests to keep data synced, if you have a tight/hot loop that is accessing a lot of object attributes, it's best to explicitly use the schema cache:
```python
>>> all_cards = ModelList()
>>> for project in planka.projects:
...     for board in project.boards:
...         all_cards.extend(board.cards)

>>> print(all_cards.extract('name'))
['My Card', ...]
```
Now you have an `all_cards` model list that contains the endpoints and a cache of all cards in your visible projects.

If you don't need the models at the end of the loop, you can instead just capture values:
```python
>>> card_due_dates = []
>>> for project in planka.projects:
...     for board in project.boards:
...         card_due_dates.extend(board.cards.extract('name', 'dueDate'))

>>> print(card_due_dates)
[('My Card', '2026...'), ('Other Card', None), ...]
```

## License
This project is licensed under the AGPLv3 License - see the [LICENSE](LICENSE) file for details.

