Skip to content

Commit

Permalink
Package library (#19)
Browse files Browse the repository at this point in the history
* Add documentation: README and better test calendar.

* example/test_calendar: Add an example of YAML anchors for templates

- YAML provides & to label a node, and ways to refer to it.  This
  allows you to make one template event, and copy it many times
  changing only one property (such as the begin time.
- This adds an example of this into example/test_calendar.yaml

* Packaging using flit

* ics into requirements and update GH action with cache and flit

* Add release action

* Update readme with package info and dev workflow

* Add warning and 0.1rc1

* BSD licensing

* Update .gitignore

* Apply suggestions from code review

Co-authored-by: Stefan van der Walt <[email protected]>

* Better PyPi with token instead of personal password

* Remove unnecessary information

* Use plain pip to install dev env

Co-authored-by: Richard Darst <[email protected]>
Co-authored-by: Stefan van der Walt <[email protected]>
  • Loading branch information
3 people authored Mar 2, 2022
1 parent eb1300b commit 1464735
Show file tree
Hide file tree
Showing 11 changed files with 242 additions and 24 deletions.
31 changes: 31 additions & 0 deletions .github/workflows/release.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
name: Release

on:
push:
tags:
- '*.*'

jobs:
release:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2

- name: Setup Python 3.10
uses: actions/setup-python@v2
with:
python-version: '3.10'
architecture: 'x64'

- name: Install flit
run: pip install flit

- name: Build
run: flit build

- name: Publish to PyPI
env:
FLIT_USERNAME: __token__
FLIT_PASSWORD: ${{ secrets.PYPI_TOKEN }}
run: |
flit publish
14 changes: 11 additions & 3 deletions .github/workflows/test.yaml → .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -17,15 +17,23 @@ jobs:
with:
python-version: ${{ matrix.python-version }}

- uses: actions/cache@v2
with:
path: ~/.cache/pip
key: ${{ runner.os }}-pip-${{ hashFiles('**/requirements.txt', '**/pyproject.toml') }}
restore-keys: |
${{ runner.os }}-pip-
- name: Install dependencies
run: |
python -m pip install --upgrade pip
pip install flake8 pytest
pip install -r requirements.txt -r requirements.dev.txt
pip install -r requirements.txt
pip install flit
flit install
- name: Lint
run: pre-commit run --all-files --show-diff-on-failure --color always

- name: Test
run: |
PYTHONPATH=. pytest
pytest
29 changes: 29 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -1,2 +1,31 @@
*~
**/__pycache__
### Python template
# Byte-compiled / optimized / DLL files
__pycache__/
*.py[cod]

# Distribution / packaging
dist/

# Unit test / coverage reports
.pytest_cache/

# General
.DS_Store

# IntelliJ project files
.idea

# Environments
.env
.venv
env/
venv/
ENV/
env.bak/
venv.bak/

# Spyder project settings
.spyderproject
.spyproject
29 changes: 29 additions & 0 deletions LICENSE
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
BSD 3-Clause License

Copyright (c) 2021--2022, Scientific Python project
All rights reserved.

Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:

1. Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.

2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.

3. Neither the name of the copyright holder nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.

THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
7 changes: 0 additions & 7 deletions LICENSE.md

This file was deleted.

75 changes: 73 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,36 @@
![PyPI](https://img.shields.io/pypi/v/yaml2ics?style=for-the-badge)

# YAML to iCalendar (ics)

Please see `example/test_calendar.yaml` for example entries.
| WARNING: this project is still in beta. Beware of breaking changes! |
|---------------------------------------------------------------------|

Convert YAML files to .ics files which can be imported into other
calendar applications.

Features include:
- Converting single .yaml files, or combining multiple into one .ics
file.
- ics fields: name, summary, description, location, timezone, repeat
- Specify event start+end or start+duration
- Recurring events (basic support)
- All-day events
- Timezone specification (default or per-event)

## Installation

```
pip install yaml2ics
```

**Note:** due to a pending release of a dependency (`ics-py`), an additional
step is required:

```
pip install -r requirements.txt
```

## Usage

To produce a calendar from a list of events:

Expand All @@ -14,8 +44,49 @@ To combine lists of events in to a calendar:
python yaml2ics.py example/test_calendar.yaml example/another_calendar.yaml
```

## Syntax

Please see `example/test_calendar.yaml` for a full demo including
explanations. Below is a minimal template that shows the basic idea:

```yaml
name: Calendar Name
timezone: Europe/Helsinki # default timezone for events, optional

events:
- summary: The event title
begin: 2021-09-21 15:00:00
duration:
minutes: 30
location: |
https://meet.jit.si/example
description: |
In this meeting we will ...
```
## Contributing
Contributions are welcomed! This project is still in active development
and should be considered beta.
To install the development version, fork the source of the project and make an
editable install:
```
pip install -r requirements.txt # temporary workaround for ics-py
pip install -e ".[test]"
```

To test:

```
PYTHONPATH=. pytest
pytest
```

[black](https://github.com/psf/black) and other linters are used to auto-format
files (and enforced by CI). To install the git hooks, use `pre-commit install`.
To run the tests/auto-formatting manually, use `pre-commit run
--all-files`.

Releases are automatically pushed on PyPi by the CI when pushing a tag
following `*.*`.
28 changes: 21 additions & 7 deletions example/test_calendar.yaml
Original file line number Diff line number Diff line change
@@ -1,20 +1,34 @@
name: Test Calendar
timezone: America/Los_Angeles
# # Include another list of events in with these. Note you can also do
# # this from the command line.
# include:
# - other_calender.ics

events:
- summary: Event of the Century
begin: 2021-09-21 15:00:00
end: 2021-09-21 15:30:00
description: |
Meet the team on the northern side of the field.
- summary: Half-an-hour meeting
begin: 2021-09-23 15:00:00 -07:00
begin: 2021-09-21 15:00:00 # uses default timezone above
duration:
minutes: 30
description: |
Meet the team on the northern side of the field.
location: |
Office 224, Monolith Bldg, Office Block C
- summary: Half-an-hour meeting
begin: 2021-09-23 15:00:00 -07:00 # explicit timezone offset
end: 2021-09-23 15:30:00 -07:00 # explicit timezone offset

- summary: Recurring event
begin: 2022-02-21 15:00:00
duration: {minutes: 60}
repeat:
interval:
# seconds, minutes, hours, days, weeks, months, years
weeks: 1
until: 2022-12-31 # required

# All-day event
- summary: Earth Day
begin: 2021-04-22
url: https://earthday.org
Expand Down
37 changes: 37 additions & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
[build-system]
requires = ["flit_core >=3.3,<4"]
build-backend = "flit_core.buildapi"

[project]
name = "yaml2ics"
version = "0.1rc1"
requires-python = ">=3.8"
authors = [{name = "The Scientific Python Group"}]
readme = "README.md"
license = {file = "LICENSE"}
classifiers = ["License :: OSI Approved :: BSD License"]
dynamic = ["description"]

dependencies = [
# "ics >=0.8",
"python-dateutil",
"pyyaml",
]

[project.optional-dependencies]
test = [
"pytest",
"black",
"pre-commit",
"flake8"
]

[project.scripts]
yaml2ics = "yaml2ics:main"

[project.urls]
Home = "https://github.com/scientific-python/yaml2ics"
Source = "https://github.com/scientific-python/yaml2ics"

[tool.flit.sdist]
exclude = ["tests/*"]
2 changes: 0 additions & 2 deletions requirements.dev.txt

This file was deleted.

2 changes: 0 additions & 2 deletions requirements.txt
Original file line number Diff line number Diff line change
@@ -1,3 +1 @@
https://github.com/ics-py/ics-py/archive/refs/heads/main.zip
python-dateutil
pyyaml
12 changes: 11 additions & 1 deletion yaml2ics.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,9 @@
"""
yaml2ics
========
CLI to convert yaml into ics.
"""
import os
import sys
from datetime import datetime, tzinfo
Expand Down Expand Up @@ -135,7 +141,7 @@ def files_to_calendar(files: list) -> ics.Calendar:
return calendar


if __name__ == "__main__":
def main():
if len(sys.argv) < 2:
print("Usage: yaml2ics.py FILE1.yaml FILE2.yaml ...")
sys.exit(-1)
Expand All @@ -149,3 +155,7 @@ def files_to_calendar(files: list) -> ics.Calendar:
calendar = files_to_calendar(files)

print(calendar.serialize())


if __name__ == "__main__":
main()

0 comments on commit 1464735

Please sign in to comment.