nyc311calendar
Python library for asynchronously fetching data for NYC school closures, trash collection holidays, and alternate side parking suspensions.
Uses the NYC 311 Public API. Built to drive the Home Assistant NYC 311 Public Services Calendar add-in.
## Warning
This is an alpha release. Expect breaking changes.
I take no responsibility for parking tickets, overflowing trash cans, kids stranded at bus stops or missing exams, etc. 🤷🏼♂️
Use at your own risk.
## Usage
### First, install via pip
```bash
pip install nyc311calendar
```
### Then, get an API key
An NYC API Portal developer account is required to use this library.
1. Sign up at https://api-portal.nyc.gov/signup/.
2. Log in, then subscribe to the "NYC 311 Public Developers" product at https://api-portal.nyc.gov/products?api=nyc-311-public-api. This subscription unlocks the calendar product.
3. Get your API key at https://api-portal.nyc.gov/developer. Either key (primary/secondary) will work.
### Finally, get data
```python
# Import modules
from nyc311calendar import CalendarType, NYC311API
# Instantiate class
calendar = NYC311API(session, API_KEY)
# Fetch calendar
resp = await calendar.get_calendar()
```
There's more to this. You'll need to provide an aiohttp ClientSession object. See the examples folder for guidance.
### Constants
This library converts strings in the source API to constants wherever sensible and uses these constants everywhere (even as dictionary keys). That is, `"status": "CLOSED"` in the source API is represented as `'status_id': }` in this library, where Status is an enum in the CivCalNYC class.
Constants are defined for:
1. Public services in `services.ServiceType`.
2. Service statuses in `services.StandardizedStatusTypes`. The source API is a mess of poorly named statuses. The StandardizedStatusTypes attempts to make sense of the underlying statuses.
3. Calendar types in `CalendarType`. See below for more info on calendar types.
### Calendar Day Entries
Data for each calendar day is returned in a `CalendarDayEntry` dataclass.
### Calendar Types
nyc311calendar can return data in several formats, each defined in the `CalendarType` enum:
#### Quarter Ahead
The Quarter Ahead calendar type returns all statuses for all services for 90 days starting on the day before the API request was made. The response dict has two sub-dicts keyed by calendar date and service. This is essentially a constant-ized dump from the source API. The example below is truncated to save space, showing two of 90 days.
```python
async with aiohttp.ClientSession() as session:
calendar = NYC311API(session, API_KEY)
resp = await calendar.get_calendar(
calendars=[CalendarType.QUARTER_AHEAD], scrub=True
)
```
```python
{
: {
: {
datetime.date(2021, 12, 31): {
: CalendarDayEntry(...),
: CalendarDayEntry(...),
: CalendarDayEntry(...)
},
datetime.date(2022, 1, 1): {
: CalendarDayEntry(...),
: CalendarDayEntry(...),
: CalendarDayEntry(...)
}
},
: {
: {
datetime.date(2021, 12, 31): CalendarDayEntry(...),
datetime.date(2022, 1, 1): CalendarDayEntry(...)
},
: {
datetime.date(2021, 12, 31): CalendarDayEntry(...),
datetime.date(2022, 1, 1): CalendarDayEntry(...)
},
: {
datetime.date(2021, 12, 31): CalendarDayEntry(...),
datetime.date(2022, 1, 1): CalendarDayEntry(...)
}
}
}
}
```
#### Week Ahead
The Week Ahead calendar type returns all statuses for all services for 8 days starting on the day before the API request was made. The response dict is keyed by number of days relative to today. This is useful for showing a calendar of the week ahead (and yesterday, just in case you forgot to move your car). The example below is truncated to save space, showing three of 90 days.
```python
async with aiohttp.ClientSession() as session:
calendar = NYC311API(session, API_KEY)
resp = await calendar.get_calendar(
calendars=[CalendarType.WEEK_AHEAD], scrub=True
)
```
```python
{
: {
-1: {
'date': datetime.date(2021, 12, 23),
'services': {
: CalendarDayEntry(...),
: CalendarDayEntry(...),
: CalendarDayEntry(...)
},
0: {
'date': datetime.date(2021, 12, 24),
'services': {
: CalendarDayEntry(...),
: CalendarDayEntry(...),
: CalendarDayEntry(...)
},
1: {
'date': datetime.date(2021, 12, 25),
'services': {
: CalendarDayEntry(...),
: CalendarDayEntry(...),
: CalendarDayEntry(...)
}
}
}
```
#### Next Exceptions
The Next Exceptions calendar type returns the next date on which there is a service exception for either of the three covered services. The response dict is keyed by service type. This is useful when you're not interested in normal operations and only want to know, say, when the next school closure is. The example below shows the full response.
Note that exceptions include things like holidays, snow days, half days, and winter break.
```python
async with aiohttp.ClientSession() as session:
calendar = NYC311API(session, API_KEY)
resp = await calendar.get_calendar(
calendars=[CalendarType.NEXT_EXCEPTIONS], scrub=True
)
```
```python
{
: {
: CalendarDayEntry(...),
: CalendarDayEntry(...),
: CalendarDayEntry(...)
}
}
```